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 131 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
142 __ stop("stop-at"); | 142 __ stop("stop-at"); |
143 } | 143 } |
144 #endif | 144 #endif |
145 | 145 |
146 // Strict mode functions and builtins need to replace the receiver | 146 // Strict mode functions and builtins need to replace the receiver |
147 // with undefined when called as functions (without an explicit | 147 // with undefined when called as functions (without an explicit |
148 // receiver object). r5 is zero for method calls and non-zero for | 148 // receiver object). r5 is zero for method calls and non-zero for |
149 // function calls. | 149 // function calls. |
150 if (!info->is_classic_mode() || info->is_native()) { | 150 if (!info->is_classic_mode() || info->is_native()) { |
151 Label ok; | 151 Label ok; |
152 __ cmp(r5, Operand(0)); | 152 __ cmp(r5, Operand::Zero()); |
153 __ b(eq, &ok); | 153 __ b(eq, &ok); |
154 int receiver_offset = info->scope()->num_parameters() * kPointerSize; | 154 int receiver_offset = info->scope()->num_parameters() * kPointerSize; |
155 __ LoadRoot(r2, Heap::kUndefinedValueRootIndex); | 155 __ LoadRoot(r2, Heap::kUndefinedValueRootIndex); |
156 __ str(r2, MemOperand(sp, receiver_offset)); | 156 __ str(r2, MemOperand(sp, receiver_offset)); |
157 __ bind(&ok); | 157 __ bind(&ok); |
158 } | 158 } |
159 | 159 |
160 // Open a frame scope to indicate that there is a frame on the stack. The | 160 // Open a frame scope to indicate that there is a frame on the stack. The |
161 // MANUAL indicates that the scope shouldn't actually generate code to set up | 161 // MANUAL indicates that the scope shouldn't actually generate code to set up |
162 // the frame (that is done below). | 162 // the frame (that is done below). |
(...skipping 869 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1032 __ b(clause->body_target()); | 1032 __ b(clause->body_target()); |
1033 __ bind(&slow_case); | 1033 __ bind(&slow_case); |
1034 } | 1034 } |
1035 | 1035 |
1036 // Record position before stub call for type feedback. | 1036 // Record position before stub call for type feedback. |
1037 SetSourcePosition(clause->position()); | 1037 SetSourcePosition(clause->position()); |
1038 Handle<Code> ic = CompareIC::GetUninitialized(Token::EQ_STRICT); | 1038 Handle<Code> ic = CompareIC::GetUninitialized(Token::EQ_STRICT); |
1039 CallIC(ic, RelocInfo::CODE_TARGET, clause->CompareId()); | 1039 CallIC(ic, RelocInfo::CODE_TARGET, clause->CompareId()); |
1040 patch_site.EmitPatchInfo(); | 1040 patch_site.EmitPatchInfo(); |
1041 | 1041 |
1042 __ cmp(r0, Operand(0)); | 1042 __ cmp(r0, Operand::Zero()); |
1043 __ b(ne, &next_test); | 1043 __ b(ne, &next_test); |
1044 __ Drop(1); // Switch value is no longer needed. | 1044 __ Drop(1); // Switch value is no longer needed. |
1045 __ b(clause->body_target()); | 1045 __ b(clause->body_target()); |
1046 } | 1046 } |
1047 | 1047 |
1048 // Discard the test value and jump to the default if present, otherwise to | 1048 // Discard the test value and jump to the default if present, otherwise to |
1049 // the end of the statement. | 1049 // the end of the statement. |
1050 __ bind(&next_test); | 1050 __ bind(&next_test); |
1051 __ Drop(1); // Switch value is no longer needed. | 1051 __ Drop(1); // Switch value is no longer needed. |
1052 if (default_clause == NULL) { | 1052 if (default_clause == NULL) { |
(...skipping 931 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1984 __ sub(scratch1, left, Operand(right), SetCC); | 1984 __ sub(scratch1, left, Operand(right), SetCC); |
1985 __ b(vs, &stub_call); | 1985 __ b(vs, &stub_call); |
1986 __ mov(right, scratch1); | 1986 __ mov(right, scratch1); |
1987 break; | 1987 break; |
1988 case Token::MUL: { | 1988 case Token::MUL: { |
1989 __ SmiUntag(ip, right); | 1989 __ SmiUntag(ip, right); |
1990 __ smull(scratch1, scratch2, left, ip); | 1990 __ smull(scratch1, scratch2, left, ip); |
1991 __ mov(ip, Operand(scratch1, ASR, 31)); | 1991 __ mov(ip, Operand(scratch1, ASR, 31)); |
1992 __ cmp(ip, Operand(scratch2)); | 1992 __ cmp(ip, Operand(scratch2)); |
1993 __ b(ne, &stub_call); | 1993 __ b(ne, &stub_call); |
1994 __ cmp(scratch1, Operand(0)); | 1994 __ cmp(scratch1, Operand::Zero()); |
1995 __ mov(right, Operand(scratch1), LeaveCC, ne); | 1995 __ mov(right, Operand(scratch1), LeaveCC, ne); |
1996 __ b(ne, &done); | 1996 __ b(ne, &done); |
1997 __ add(scratch2, right, Operand(left), SetCC); | 1997 __ add(scratch2, right, Operand(left), SetCC); |
1998 __ mov(right, Operand(Smi::FromInt(0)), LeaveCC, pl); | 1998 __ mov(right, Operand(Smi::FromInt(0)), LeaveCC, pl); |
1999 __ b(mi, &stub_call); | 1999 __ b(mi, &stub_call); |
2000 break; | 2000 break; |
2001 } | 2001 } |
2002 case Token::BIT_OR: | 2002 case Token::BIT_OR: |
2003 __ orr(right, left, Operand(right)); | 2003 __ orr(right, left, Operand(right)); |
2004 break; | 2004 break; |
(...skipping 675 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2680 __ cmp(r2, ip); | 2680 __ cmp(r2, ip); |
2681 __ b(eq, if_false); | 2681 __ b(eq, if_false); |
2682 | 2682 |
2683 // Look for valueOf symbol in the descriptor array, and indicate false if | 2683 // Look for valueOf symbol in the descriptor array, and indicate false if |
2684 // found. Since we omit an enumeration index check, if it is added via a | 2684 // found. Since we omit an enumeration index check, if it is added via a |
2685 // transition that shares its descriptor array, this is a false positive. | 2685 // transition that shares its descriptor array, this is a false positive. |
2686 Label entry, loop, done; | 2686 Label entry, loop, done; |
2687 | 2687 |
2688 // Skip loop if no descriptors are valid. | 2688 // Skip loop if no descriptors are valid. |
2689 __ NumberOfOwnDescriptors(r3, r1); | 2689 __ NumberOfOwnDescriptors(r3, r1); |
2690 __ cmp(r3, Operand(0)); | 2690 __ cmp(r3, Operand::Zero()); |
2691 __ b(eq, &done); | 2691 __ b(eq, &done); |
2692 | 2692 |
2693 __ LoadInstanceDescriptors(r1, r4); | 2693 __ LoadInstanceDescriptors(r1, r4); |
2694 // r4: descriptor array. | 2694 // r4: descriptor array. |
2695 // r3: valid entries in the descriptor array. | 2695 // r3: valid entries in the descriptor array. |
2696 STATIC_ASSERT(kSmiTag == 0); | 2696 STATIC_ASSERT(kSmiTag == 0); |
2697 STATIC_ASSERT(kSmiTagSize == 1); | 2697 STATIC_ASSERT(kSmiTagSize == 1); |
2698 STATIC_ASSERT(kPointerSize == 4); | 2698 STATIC_ASSERT(kPointerSize == 4); |
2699 __ mov(ip, Operand(DescriptorArray::kDescriptorSize)); | 2699 __ mov(ip, Operand(DescriptorArray::kDescriptorSize)); |
2700 __ mul(r3, r3, ip); | 2700 __ mul(r3, r3, ip); |
(...skipping 310 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3011 __ CallCFunction(ExternalReference::random_uint32_function(isolate()), 1); | 3011 __ CallCFunction(ExternalReference::random_uint32_function(isolate()), 1); |
3012 | 3012 |
3013 CpuFeatures::Scope scope(VFP2); | 3013 CpuFeatures::Scope scope(VFP2); |
3014 // 0x41300000 is the top half of 1.0 x 2^20 as a double. | 3014 // 0x41300000 is the top half of 1.0 x 2^20 as a double. |
3015 // Create this constant using mov/orr to avoid PC relative load. | 3015 // Create this constant using mov/orr to avoid PC relative load. |
3016 __ mov(r1, Operand(0x41000000)); | 3016 __ mov(r1, Operand(0x41000000)); |
3017 __ orr(r1, r1, Operand(0x300000)); | 3017 __ orr(r1, r1, Operand(0x300000)); |
3018 // Move 0x41300000xxxxxxxx (x = random bits) to VFP. | 3018 // Move 0x41300000xxxxxxxx (x = random bits) to VFP. |
3019 __ vmov(d7, r0, r1); | 3019 __ vmov(d7, r0, r1); |
3020 // Move 0x4130000000000000 to VFP. | 3020 // Move 0x4130000000000000 to VFP. |
3021 __ mov(r0, Operand(0, RelocInfo::NONE32)); | 3021 __ mov(r0, Operand::Zero()); |
3022 __ vmov(d8, r0, r1); | 3022 __ vmov(d8, r0, r1); |
3023 // Subtract and store the result in the heap number. | 3023 // Subtract and store the result in the heap number. |
3024 __ vsub(d7, d7, d8); | 3024 __ vsub(d7, d7, d8); |
3025 __ sub(r0, r4, Operand(kHeapObjectTag)); | 3025 __ sub(r0, r4, Operand(kHeapObjectTag)); |
3026 __ vstr(d7, r0, HeapNumber::kValueOffset); | 3026 __ vstr(d7, r0, HeapNumber::kValueOffset); |
3027 __ mov(r0, r4); | 3027 __ mov(r0, r4); |
3028 } else { | 3028 } else { |
3029 __ PrepareCallCFunction(2, r0); | 3029 __ PrepareCallCFunction(2, r0); |
3030 __ ldr(r1, | 3030 __ ldr(r1, |
3031 ContextOperand(context_register(), Context::GLOBAL_OBJECT_INDEX)); | 3031 ContextOperand(context_register(), Context::GLOBAL_OBJECT_INDEX)); |
(...skipping 595 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3627 | 3627 |
3628 __ bind(&non_trivial_array); | 3628 __ bind(&non_trivial_array); |
3629 | 3629 |
3630 // Get the FixedArray containing array's elements. | 3630 // Get the FixedArray containing array's elements. |
3631 elements = array; | 3631 elements = array; |
3632 __ ldr(elements, FieldMemOperand(array, JSArray::kElementsOffset)); | 3632 __ ldr(elements, FieldMemOperand(array, JSArray::kElementsOffset)); |
3633 array = no_reg; // End of array's live range. | 3633 array = no_reg; // End of array's live range. |
3634 | 3634 |
3635 // Check that all array elements are sequential ASCII strings, and | 3635 // Check that all array elements are sequential ASCII strings, and |
3636 // accumulate the sum of their lengths, as a smi-encoded value. | 3636 // accumulate the sum of their lengths, as a smi-encoded value. |
3637 __ mov(string_length, Operand(0)); | 3637 __ mov(string_length, Operand::Zero()); |
3638 __ add(element, | 3638 __ add(element, |
3639 elements, Operand(FixedArray::kHeaderSize - kHeapObjectTag)); | 3639 elements, Operand(FixedArray::kHeaderSize - kHeapObjectTag)); |
3640 __ add(elements_end, element, Operand(array_length, LSL, kPointerSizeLog2)); | 3640 __ add(elements_end, element, Operand(array_length, LSL, kPointerSizeLog2)); |
3641 // Loop condition: while (element < elements_end). | 3641 // Loop condition: while (element < elements_end). |
3642 // Live values in registers: | 3642 // Live values in registers: |
3643 // elements: Fixed array of strings. | 3643 // elements: Fixed array of strings. |
3644 // array_length: Length of the fixed array of strings (not smi) | 3644 // array_length: Length of the fixed array of strings (not smi) |
3645 // separator: Separator string | 3645 // separator: Separator string |
3646 // string_length: Accumulated sum of string lengths (smi). | 3646 // string_length: Accumulated sum of string lengths (smi). |
3647 // element: Current array element. | 3647 // element: Current array element. |
3648 // elements_end: Array end. | 3648 // elements_end: Array end. |
3649 if (generate_debug_code_) { | 3649 if (generate_debug_code_) { |
3650 __ cmp(array_length, Operand(0)); | 3650 __ cmp(array_length, Operand::Zero()); |
3651 __ Assert(gt, "No empty arrays here in EmitFastAsciiArrayJoin"); | 3651 __ Assert(gt, "No empty arrays here in EmitFastAsciiArrayJoin"); |
3652 } | 3652 } |
3653 __ bind(&loop); | 3653 __ bind(&loop); |
3654 __ ldr(string, MemOperand(element, kPointerSize, PostIndex)); | 3654 __ ldr(string, MemOperand(element, kPointerSize, PostIndex)); |
3655 __ JumpIfSmi(string, &bailout); | 3655 __ JumpIfSmi(string, &bailout); |
3656 __ ldr(scratch1, FieldMemOperand(string, HeapObject::kMapOffset)); | 3656 __ ldr(scratch1, FieldMemOperand(string, HeapObject::kMapOffset)); |
3657 __ ldrb(scratch1, FieldMemOperand(scratch1, Map::kInstanceTypeOffset)); | 3657 __ ldrb(scratch1, FieldMemOperand(scratch1, Map::kInstanceTypeOffset)); |
3658 __ JumpIfInstanceTypeIsNotSequentialAscii(scratch1, scratch2, &bailout); | 3658 __ JumpIfInstanceTypeIsNotSequentialAscii(scratch1, scratch2, &bailout); |
3659 __ ldr(scratch1, FieldMemOperand(string, SeqOneByteString::kLengthOffset)); | 3659 __ ldr(scratch1, FieldMemOperand(string, SeqOneByteString::kLengthOffset)); |
3660 __ add(string_length, string_length, Operand(scratch1), SetCC); | 3660 __ add(string_length, string_length, Operand(scratch1), SetCC); |
(...skipping 22 matching lines...) Expand all Loading... |
3683 __ JumpIfInstanceTypeIsNotSequentialAscii(scratch1, scratch2, &bailout); | 3683 __ JumpIfInstanceTypeIsNotSequentialAscii(scratch1, scratch2, &bailout); |
3684 | 3684 |
3685 // Add (separator length times array_length) - separator length to the | 3685 // Add (separator length times array_length) - separator length to the |
3686 // string_length to get the length of the result string. array_length is not | 3686 // string_length to get the length of the result string. array_length is not |
3687 // smi but the other values are, so the result is a smi | 3687 // smi but the other values are, so the result is a smi |
3688 __ ldr(scratch1, FieldMemOperand(separator, SeqOneByteString::kLengthOffset)); | 3688 __ ldr(scratch1, FieldMemOperand(separator, SeqOneByteString::kLengthOffset)); |
3689 __ sub(string_length, string_length, Operand(scratch1)); | 3689 __ sub(string_length, string_length, Operand(scratch1)); |
3690 __ smull(scratch2, ip, array_length, scratch1); | 3690 __ smull(scratch2, ip, array_length, scratch1); |
3691 // Check for smi overflow. No overflow if higher 33 bits of 64-bit result are | 3691 // Check for smi overflow. No overflow if higher 33 bits of 64-bit result are |
3692 // zero. | 3692 // zero. |
3693 __ cmp(ip, Operand(0)); | 3693 __ cmp(ip, Operand::Zero()); |
3694 __ b(ne, &bailout); | 3694 __ b(ne, &bailout); |
3695 __ tst(scratch2, Operand(0x80000000)); | 3695 __ tst(scratch2, Operand(0x80000000)); |
3696 __ b(ne, &bailout); | 3696 __ b(ne, &bailout); |
3697 __ add(string_length, string_length, Operand(scratch2), SetCC); | 3697 __ add(string_length, string_length, Operand(scratch2), SetCC); |
3698 __ b(vs, &bailout); | 3698 __ b(vs, &bailout); |
3699 __ SmiUntag(string_length); | 3699 __ SmiUntag(string_length); |
3700 | 3700 |
3701 // Get first element in the array to free up the elements register to be used | 3701 // Get first element in the array to free up the elements register to be used |
3702 // for the result. | 3702 // for the result. |
3703 __ add(element, | 3703 __ add(element, |
(...skipping 646 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4350 Split(cond, if_true, if_false, NULL); | 4350 Split(cond, if_true, if_false, NULL); |
4351 __ bind(&slow_case); | 4351 __ bind(&slow_case); |
4352 } | 4352 } |
4353 | 4353 |
4354 // Record position and call the compare IC. | 4354 // Record position and call the compare IC. |
4355 SetSourcePosition(expr->position()); | 4355 SetSourcePosition(expr->position()); |
4356 Handle<Code> ic = CompareIC::GetUninitialized(op); | 4356 Handle<Code> ic = CompareIC::GetUninitialized(op); |
4357 CallIC(ic, RelocInfo::CODE_TARGET, expr->CompareOperationFeedbackId()); | 4357 CallIC(ic, RelocInfo::CODE_TARGET, expr->CompareOperationFeedbackId()); |
4358 patch_site.EmitPatchInfo(); | 4358 patch_site.EmitPatchInfo(); |
4359 PrepareForBailoutBeforeSplit(expr, true, if_true, if_false); | 4359 PrepareForBailoutBeforeSplit(expr, true, if_true, if_false); |
4360 __ cmp(r0, Operand(0)); | 4360 __ cmp(r0, Operand::Zero()); |
4361 Split(cond, if_true, if_false, fall_through); | 4361 Split(cond, if_true, if_false, fall_through); |
4362 } | 4362 } |
4363 } | 4363 } |
4364 | 4364 |
4365 // Convert the result of the comparison into one expected for this | 4365 // Convert the result of the comparison into one expected for this |
4366 // expression's context. | 4366 // expression's context. |
4367 context()->Plug(if_true, if_false); | 4367 context()->Plug(if_true, if_false); |
4368 } | 4368 } |
4369 | 4369 |
4370 | 4370 |
(...skipping 182 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4553 *context_length = 0; | 4553 *context_length = 0; |
4554 return previous_; | 4554 return previous_; |
4555 } | 4555 } |
4556 | 4556 |
4557 | 4557 |
4558 #undef __ | 4558 #undef __ |
4559 | 4559 |
4560 } } // namespace v8::internal | 4560 } } // namespace v8::internal |
4561 | 4561 |
4562 #endif // V8_TARGET_ARCH_ARM | 4562 #endif // V8_TARGET_ARCH_ARM |
OLD | NEW |