Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
| 4 // met: | 4 // met: |
| 5 // | 5 // |
| 6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
| 7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
| 8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
| 9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
| 10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
| (...skipping 2678 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 2689 __ LoadRoot(ip, Heap::kFixedArrayMapRootIndex); | 2689 __ LoadRoot(ip, Heap::kFixedArrayMapRootIndex); |
| 2690 __ cmp(scratch, ip); | 2690 __ cmp(scratch, ip); |
| 2691 __ b(eq, &done); | 2691 __ b(eq, &done); |
| 2692 __ LoadRoot(ip, Heap::kFixedCOWArrayMapRootIndex); | 2692 __ LoadRoot(ip, Heap::kFixedCOWArrayMapRootIndex); |
| 2693 __ cmp(scratch, ip); | 2693 __ cmp(scratch, ip); |
| 2694 __ b(eq, &done); | 2694 __ b(eq, &done); |
| 2695 // |scratch| still contains |input|'s map. | 2695 // |scratch| still contains |input|'s map. |
| 2696 __ ldr(scratch, FieldMemOperand(scratch, Map::kBitField2Offset)); | 2696 __ ldr(scratch, FieldMemOperand(scratch, Map::kBitField2Offset)); |
| 2697 __ ubfx(scratch, scratch, Map::kElementsKindShift, | 2697 __ ubfx(scratch, scratch, Map::kElementsKindShift, |
| 2698 Map::kElementsKindBitCount); | 2698 Map::kElementsKindBitCount); |
| 2699 __ cmp(scratch, Operand(FAST_ELEMENTS)); | 2699 __ cmp(scratch, Operand(GetInitialFastElementsKind())); |
| 2700 __ b(eq, &done); | 2700 __ b(lt, &fail); |
| 2701 __ cmp(scratch, Operand(TERMINAL_FAST_ELEMENTS_KIND)); | |
| 2702 __ b(le, &done); | |
| 2701 __ cmp(scratch, Operand(FIRST_EXTERNAL_ARRAY_ELEMENTS_KIND)); | 2703 __ cmp(scratch, Operand(FIRST_EXTERNAL_ARRAY_ELEMENTS_KIND)); |
| 2702 __ b(lt, &fail); | 2704 __ b(lt, &fail); |
| 2703 __ cmp(scratch, Operand(LAST_EXTERNAL_ARRAY_ELEMENTS_KIND)); | 2705 __ cmp(scratch, Operand(LAST_EXTERNAL_ARRAY_ELEMENTS_KIND)); |
| 2704 __ b(le, &done); | 2706 __ b(le, &done); |
| 2705 __ bind(&fail); | 2707 __ bind(&fail); |
| 2706 __ Abort("Check for fast or external elements failed."); | 2708 __ Abort("Check for fast or external elements failed."); |
| 2707 __ bind(&done); | 2709 __ bind(&done); |
| 2708 } | 2710 } |
| 2709 } | 2711 } |
| 2710 | 2712 |
| (...skipping 67 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 2778 Operand operand = key_is_constant | 2780 Operand operand = key_is_constant |
| 2779 ? Operand(constant_key * (1 << shift_size) + | 2781 ? Operand(constant_key * (1 << shift_size) + |
| 2780 FixedDoubleArray::kHeaderSize - kHeapObjectTag) | 2782 FixedDoubleArray::kHeaderSize - kHeapObjectTag) |
| 2781 : Operand(key, LSL, shift_size); | 2783 : Operand(key, LSL, shift_size); |
| 2782 __ add(elements, elements, operand); | 2784 __ add(elements, elements, operand); |
| 2783 if (!key_is_constant) { | 2785 if (!key_is_constant) { |
| 2784 __ add(elements, elements, | 2786 __ add(elements, elements, |
| 2785 Operand(FixedDoubleArray::kHeaderSize - kHeapObjectTag)); | 2787 Operand(FixedDoubleArray::kHeaderSize - kHeapObjectTag)); |
| 2786 } | 2788 } |
| 2787 | 2789 |
| 2788 __ ldr(scratch, MemOperand(elements, sizeof(kHoleNanLower32))); | 2790 if (instr->hydrogen()->RequiresHoleCheck()) { |
| 2789 __ cmp(scratch, Operand(kHoleNanUpper32)); | 2791 __ ldr(scratch, MemOperand(elements, sizeof(kHoleNanLower32))); |
| 2790 DeoptimizeIf(eq, instr->environment()); | 2792 __ cmp(scratch, Operand(kHoleNanUpper32)); |
| 2793 DeoptimizeIf(eq, instr->environment()); | |
| 2794 } | |
| 2791 | 2795 |
| 2792 __ vldr(result, elements, 0); | 2796 __ vldr(result, elements, 0); |
| 2793 } | 2797 } |
| 2794 | 2798 |
| 2795 | 2799 |
| 2796 void LCodeGen::DoLoadKeyedSpecializedArrayElement( | 2800 void LCodeGen::DoLoadKeyedSpecializedArrayElement( |
| 2797 LLoadKeyedSpecializedArrayElement* instr) { | 2801 LLoadKeyedSpecializedArrayElement* instr) { |
| 2798 Register external_pointer = ToRegister(instr->external_pointer()); | 2802 Register external_pointer = ToRegister(instr->external_pointer()); |
| 2799 Register key = no_reg; | 2803 Register key = no_reg; |
| 2800 ElementsKind elements_kind = instr->elements_kind(); | 2804 ElementsKind elements_kind = instr->elements_kind(); |
| (...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 2849 case EXTERNAL_UNSIGNED_INT_ELEMENTS: | 2853 case EXTERNAL_UNSIGNED_INT_ELEMENTS: |
| 2850 __ ldr(result, mem_operand); | 2854 __ ldr(result, mem_operand); |
| 2851 __ cmp(result, Operand(0x80000000)); | 2855 __ cmp(result, Operand(0x80000000)); |
| 2852 // TODO(danno): we could be more clever here, perhaps having a special | 2856 // TODO(danno): we could be more clever here, perhaps having a special |
| 2853 // version of the stub that detects if the overflow case actually | 2857 // version of the stub that detects if the overflow case actually |
| 2854 // happens, and generate code that returns a double rather than int. | 2858 // happens, and generate code that returns a double rather than int. |
| 2855 DeoptimizeIf(cs, instr->environment()); | 2859 DeoptimizeIf(cs, instr->environment()); |
| 2856 break; | 2860 break; |
| 2857 case EXTERNAL_FLOAT_ELEMENTS: | 2861 case EXTERNAL_FLOAT_ELEMENTS: |
| 2858 case EXTERNAL_DOUBLE_ELEMENTS: | 2862 case EXTERNAL_DOUBLE_ELEMENTS: |
| 2863 case FAST_HOLEY_DOUBLE_ELEMENTS: | |
| 2864 case FAST_HOLEY_ELEMENTS: | |
| 2865 case FAST_HOLEY_SMI_ELEMENTS: | |
| 2859 case FAST_DOUBLE_ELEMENTS: | 2866 case FAST_DOUBLE_ELEMENTS: |
| 2860 case FAST_ELEMENTS: | 2867 case FAST_ELEMENTS: |
| 2861 case FAST_SMI_ONLY_ELEMENTS: | 2868 case FAST_SMI_ELEMENTS: |
| 2862 case DICTIONARY_ELEMENTS: | 2869 case DICTIONARY_ELEMENTS: |
| 2863 case NON_STRICT_ARGUMENTS_ELEMENTS: | 2870 case NON_STRICT_ARGUMENTS_ELEMENTS: |
| 2864 UNREACHABLE(); | 2871 UNREACHABLE(); |
| 2865 break; | 2872 break; |
| 2866 } | 2873 } |
| 2867 } | 2874 } |
| 2868 } | 2875 } |
| 2869 | 2876 |
| 2870 | 2877 |
| 2871 void LCodeGen::DoLoadKeyedGeneric(LLoadKeyedGeneric* instr) { | 2878 void LCodeGen::DoLoadKeyedGeneric(LLoadKeyedGeneric* instr) { |
| (...skipping 968 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 3840 __ strh(value, mem_operand); | 3847 __ strh(value, mem_operand); |
| 3841 break; | 3848 break; |
| 3842 case EXTERNAL_INT_ELEMENTS: | 3849 case EXTERNAL_INT_ELEMENTS: |
| 3843 case EXTERNAL_UNSIGNED_INT_ELEMENTS: | 3850 case EXTERNAL_UNSIGNED_INT_ELEMENTS: |
| 3844 __ str(value, mem_operand); | 3851 __ str(value, mem_operand); |
| 3845 break; | 3852 break; |
| 3846 case EXTERNAL_FLOAT_ELEMENTS: | 3853 case EXTERNAL_FLOAT_ELEMENTS: |
| 3847 case EXTERNAL_DOUBLE_ELEMENTS: | 3854 case EXTERNAL_DOUBLE_ELEMENTS: |
| 3848 case FAST_DOUBLE_ELEMENTS: | 3855 case FAST_DOUBLE_ELEMENTS: |
| 3849 case FAST_ELEMENTS: | 3856 case FAST_ELEMENTS: |
| 3850 case FAST_SMI_ONLY_ELEMENTS: | 3857 case FAST_SMI_ELEMENTS: |
| 3858 case FAST_HOLEY_DOUBLE_ELEMENTS: | |
| 3859 case FAST_HOLEY_ELEMENTS: | |
| 3860 case FAST_HOLEY_SMI_ELEMENTS: | |
| 3851 case DICTIONARY_ELEMENTS: | 3861 case DICTIONARY_ELEMENTS: |
| 3852 case NON_STRICT_ARGUMENTS_ELEMENTS: | 3862 case NON_STRICT_ARGUMENTS_ELEMENTS: |
| 3853 UNREACHABLE(); | 3863 UNREACHABLE(); |
| 3854 break; | 3864 break; |
| 3855 } | 3865 } |
| 3856 } | 3866 } |
| 3857 } | 3867 } |
| 3858 | 3868 |
| 3859 | 3869 |
| 3860 void LCodeGen::DoStoreKeyedGeneric(LStoreKeyedGeneric* instr) { | 3870 void LCodeGen::DoStoreKeyedGeneric(LStoreKeyedGeneric* instr) { |
| (...skipping 16 matching lines...) Expand all Loading... | |
| 3877 Handle<Map> from_map = instr->original_map(); | 3887 Handle<Map> from_map = instr->original_map(); |
| 3878 Handle<Map> to_map = instr->transitioned_map(); | 3888 Handle<Map> to_map = instr->transitioned_map(); |
| 3879 ElementsKind from_kind = from_map->elements_kind(); | 3889 ElementsKind from_kind = from_map->elements_kind(); |
| 3880 ElementsKind to_kind = to_map->elements_kind(); | 3890 ElementsKind to_kind = to_map->elements_kind(); |
| 3881 | 3891 |
| 3882 Label not_applicable; | 3892 Label not_applicable; |
| 3883 __ ldr(scratch, FieldMemOperand(object_reg, HeapObject::kMapOffset)); | 3893 __ ldr(scratch, FieldMemOperand(object_reg, HeapObject::kMapOffset)); |
| 3884 __ cmp(scratch, Operand(from_map)); | 3894 __ cmp(scratch, Operand(from_map)); |
| 3885 __ b(ne, ¬_applicable); | 3895 __ b(ne, ¬_applicable); |
| 3886 __ mov(new_map_reg, Operand(to_map)); | 3896 __ mov(new_map_reg, Operand(to_map)); |
| 3887 if (from_kind == FAST_SMI_ONLY_ELEMENTS && to_kind == FAST_ELEMENTS) { | 3897 |
| 3898 bool simple_map_change = (GetHoleyElementsKind(from_kind) == to_kind) || | |
| 3899 (IsFastSmiElementsKind(from_kind) && | |
| 3900 IsFastObjectElementsKind(to_kind)); | |
| 3901 if (simple_map_change) { | |
| 3888 __ str(new_map_reg, FieldMemOperand(object_reg, HeapObject::kMapOffset)); | 3902 __ str(new_map_reg, FieldMemOperand(object_reg, HeapObject::kMapOffset)); |
| 3889 // Write barrier. | 3903 // Write barrier. |
| 3890 __ RecordWriteField(object_reg, HeapObject::kMapOffset, new_map_reg, | 3904 __ RecordWriteField(object_reg, HeapObject::kMapOffset, new_map_reg, |
| 3891 scratch, kLRHasBeenSaved, kDontSaveFPRegs); | 3905 scratch, kLRHasBeenSaved, kDontSaveFPRegs); |
| 3892 } else if (from_kind == FAST_SMI_ONLY_ELEMENTS && | 3906 } else if (IsFastSmiElementsKind(from_kind) && |
| 3893 to_kind == FAST_DOUBLE_ELEMENTS) { | 3907 IsFastDoubleElementsKind(to_kind)) { |
| 3894 Register fixed_object_reg = ToRegister(instr->temp_reg()); | 3908 Register fixed_object_reg = ToRegister(instr->temp_reg()); |
| 3895 ASSERT(fixed_object_reg.is(r2)); | 3909 ASSERT(fixed_object_reg.is(r2)); |
| 3896 ASSERT(new_map_reg.is(r3)); | 3910 ASSERT(new_map_reg.is(r3)); |
| 3897 __ mov(fixed_object_reg, object_reg); | 3911 __ mov(fixed_object_reg, object_reg); |
| 3898 CallCode(isolate()->builtins()->TransitionElementsSmiToDouble(), | 3912 CallCode(isolate()->builtins()->TransitionElementsSmiToDouble(), |
| 3899 RelocInfo::CODE_TARGET, instr); | 3913 RelocInfo::CODE_TARGET, instr); |
| 3900 } else if (from_kind == FAST_DOUBLE_ELEMENTS && to_kind == FAST_ELEMENTS) { | 3914 } else if (IsFastDoubleElementsKind(from_kind) && |
| 3915 IsFastElementsKind(to_kind)) { | |
|
Jakob Kummerow
2012/05/22 17:36:49
IsFastObjectElementsKind(to_kind) ?
danno
2012/05/23 14:25:36
Done.
| |
| 3901 Register fixed_object_reg = ToRegister(instr->temp_reg()); | 3916 Register fixed_object_reg = ToRegister(instr->temp_reg()); |
| 3902 ASSERT(fixed_object_reg.is(r2)); | 3917 ASSERT(fixed_object_reg.is(r2)); |
| 3903 ASSERT(new_map_reg.is(r3)); | 3918 ASSERT(new_map_reg.is(r3)); |
| 3904 __ mov(fixed_object_reg, object_reg); | 3919 __ mov(fixed_object_reg, object_reg); |
| 3905 CallCode(isolate()->builtins()->TransitionElementsDoubleToObject(), | 3920 CallCode(isolate()->builtins()->TransitionElementsDoubleToObject(), |
| 3906 RelocInfo::CODE_TARGET, instr); | 3921 RelocInfo::CODE_TARGET, instr); |
| 3907 } else { | 3922 } else { |
| 3908 UNREACHABLE(); | 3923 UNREACHABLE(); |
| 3909 } | 3924 } |
| 3910 __ bind(¬_applicable); | 3925 __ bind(¬_applicable); |
| (...skipping 753 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 4664 } | 4679 } |
| 4665 | 4680 |
| 4666 | 4681 |
| 4667 void LCodeGen::DoArrayLiteral(LArrayLiteral* instr) { | 4682 void LCodeGen::DoArrayLiteral(LArrayLiteral* instr) { |
| 4668 Heap* heap = isolate()->heap(); | 4683 Heap* heap = isolate()->heap(); |
| 4669 ElementsKind boilerplate_elements_kind = | 4684 ElementsKind boilerplate_elements_kind = |
| 4670 instr->hydrogen()->boilerplate_elements_kind(); | 4685 instr->hydrogen()->boilerplate_elements_kind(); |
| 4671 | 4686 |
| 4672 // Deopt if the array literal boilerplate ElementsKind is of a type different | 4687 // Deopt if the array literal boilerplate ElementsKind is of a type different |
| 4673 // than the expected one. The check isn't necessary if the boilerplate has | 4688 // than the expected one. The check isn't necessary if the boilerplate has |
| 4674 // already been converted to FAST_ELEMENTS. | 4689 // already been converted to TERMINAL_FAST_ELEMENTS_KIND. |
| 4675 if (boilerplate_elements_kind != FAST_ELEMENTS) { | 4690 if (CanTransitionToMoreGeneralFastElementsKind( |
| 4691 boilerplate_elements_kind, true)) { | |
| 4676 __ LoadHeapObject(r1, instr->hydrogen()->boilerplate_object()); | 4692 __ LoadHeapObject(r1, instr->hydrogen()->boilerplate_object()); |
| 4677 // Load map into r2. | 4693 // Load map into r2. |
| 4678 __ ldr(r2, FieldMemOperand(r1, HeapObject::kMapOffset)); | 4694 __ ldr(r2, FieldMemOperand(r1, HeapObject::kMapOffset)); |
| 4679 // Load the map's "bit field 2". | 4695 // Load the map's "bit field 2". |
| 4680 __ ldrb(r2, FieldMemOperand(r2, Map::kBitField2Offset)); | 4696 __ ldrb(r2, FieldMemOperand(r2, Map::kBitField2Offset)); |
| 4681 // Retrieve elements_kind from bit field 2. | 4697 // Retrieve elements_kind from bit field 2. |
| 4682 __ ubfx(r2, r2, Map::kElementsKindShift, Map::kElementsKindBitCount); | 4698 __ ubfx(r2, r2, Map::kElementsKindShift, Map::kElementsKindBitCount); |
| 4683 __ cmp(r2, Operand(boilerplate_elements_kind)); | 4699 __ cmp(r2, Operand(boilerplate_elements_kind)); |
| 4684 DeoptimizeIf(ne, instr->environment()); | 4700 DeoptimizeIf(ne, instr->environment()); |
| 4685 } | 4701 } |
| (...skipping 130 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 4816 } | 4832 } |
| 4817 } | 4833 } |
| 4818 } | 4834 } |
| 4819 | 4835 |
| 4820 | 4836 |
| 4821 void LCodeGen::DoFastLiteral(LFastLiteral* instr) { | 4837 void LCodeGen::DoFastLiteral(LFastLiteral* instr) { |
| 4822 int size = instr->hydrogen()->total_size(); | 4838 int size = instr->hydrogen()->total_size(); |
| 4823 ElementsKind boilerplate_elements_kind = | 4839 ElementsKind boilerplate_elements_kind = |
| 4824 instr->hydrogen()->boilerplate()->GetElementsKind(); | 4840 instr->hydrogen()->boilerplate()->GetElementsKind(); |
| 4825 | 4841 |
| 4826 // Deopt if the literal boilerplate ElementsKind is of a type different than | 4842 // Deopt if the array literal boilerplate ElementsKind is of a type different |
| 4827 // the expected one. The check isn't necessary if the boilerplate has already | 4843 // than the expected one. The check isn't necessary if the boilerplate has |
| 4828 // been converted to FAST_ELEMENTS. | 4844 // already been converted to TERMINAL_FAST_ELEMENTS_KIND. |
| 4829 if (boilerplate_elements_kind != FAST_ELEMENTS) { | 4845 if (CanTransitionToMoreGeneralFastElementsKind( |
| 4846 boilerplate_elements_kind, true)) { | |
| 4830 __ LoadHeapObject(r1, instr->hydrogen()->boilerplate()); | 4847 __ LoadHeapObject(r1, instr->hydrogen()->boilerplate()); |
| 4831 // Load map into r2. | 4848 // Load map into r2. |
| 4832 __ ldr(r2, FieldMemOperand(r1, HeapObject::kMapOffset)); | 4849 __ ldr(r2, FieldMemOperand(r1, HeapObject::kMapOffset)); |
| 4833 // Load the map's "bit field 2". | 4850 // Load the map's "bit field 2". |
| 4834 __ ldrb(r2, FieldMemOperand(r2, Map::kBitField2Offset)); | 4851 __ ldrb(r2, FieldMemOperand(r2, Map::kBitField2Offset)); |
| 4835 // Retrieve elements_kind from bit field 2. | 4852 // Retrieve elements_kind from bit field 2. |
| 4836 __ ubfx(r2, r2, Map::kElementsKindShift, Map::kElementsKindBitCount); | 4853 __ ubfx(r2, r2, Map::kElementsKindShift, Map::kElementsKindBitCount); |
| 4837 __ cmp(r2, Operand(boilerplate_elements_kind)); | 4854 __ cmp(r2, Operand(boilerplate_elements_kind)); |
| 4838 DeoptimizeIf(ne, instr->environment()); | 4855 DeoptimizeIf(ne, instr->environment()); |
| 4839 } | 4856 } |
| (...skipping 467 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 5307 __ sub(scratch, result, Operand(index, LSL, kPointerSizeLog2 - kSmiTagSize)); | 5324 __ sub(scratch, result, Operand(index, LSL, kPointerSizeLog2 - kSmiTagSize)); |
| 5308 __ ldr(result, FieldMemOperand(scratch, | 5325 __ ldr(result, FieldMemOperand(scratch, |
| 5309 FixedArray::kHeaderSize - kPointerSize)); | 5326 FixedArray::kHeaderSize - kPointerSize)); |
| 5310 __ bind(&done); | 5327 __ bind(&done); |
| 5311 } | 5328 } |
| 5312 | 5329 |
| 5313 | 5330 |
| 5314 #undef __ | 5331 #undef __ |
| 5315 | 5332 |
| 5316 } } // namespace v8::internal | 5333 } } // namespace v8::internal |
| OLD | NEW |