OLD | NEW |
1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 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 3501 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3512 __ cvttsd2si(output_reg, input_reg); | 3512 __ cvttsd2si(output_reg, input_reg); |
3513 // Overflow is signalled with minint. | 3513 // Overflow is signalled with minint. |
3514 __ cmpl(output_reg, Immediate(0x80000000)); | 3514 __ cmpl(output_reg, Immediate(0x80000000)); |
3515 DeoptimizeIf(equal, instr->environment()); | 3515 DeoptimizeIf(equal, instr->environment()); |
3516 __ jmp(&done, Label::kNear); | 3516 __ jmp(&done, Label::kNear); |
3517 | 3517 |
3518 // Non-zero negative reaches here. | 3518 // Non-zero negative reaches here. |
3519 __ bind(&negative_sign); | 3519 __ bind(&negative_sign); |
3520 // Truncate, then compare and compensate. | 3520 // Truncate, then compare and compensate. |
3521 __ cvttsd2si(output_reg, input_reg); | 3521 __ cvttsd2si(output_reg, input_reg); |
3522 __ cvtlsi2sd(xmm_scratch, output_reg); | 3522 __ Cvtlsi2sd(xmm_scratch, output_reg); |
3523 __ ucomisd(input_reg, xmm_scratch); | 3523 __ ucomisd(input_reg, xmm_scratch); |
3524 __ j(equal, &done, Label::kNear); | 3524 __ j(equal, &done, Label::kNear); |
3525 __ subl(output_reg, Immediate(1)); | 3525 __ subl(output_reg, Immediate(1)); |
3526 DeoptimizeIf(overflow, instr->environment()); | 3526 DeoptimizeIf(overflow, instr->environment()); |
3527 | 3527 |
3528 __ bind(&done); | 3528 __ bind(&done); |
3529 } | 3529 } |
3530 } | 3530 } |
3531 | 3531 |
3532 | 3532 |
(...skipping 28 matching lines...) Expand all Loading... |
3561 // CVTTSD2SI rounds towards zero, we use ceil(x - (-0.5)) and then | 3561 // CVTTSD2SI rounds towards zero, we use ceil(x - (-0.5)) and then |
3562 // compare and compensate. | 3562 // compare and compensate. |
3563 __ movq(kScratchRegister, input_reg); // Back up input_reg. | 3563 __ movq(kScratchRegister, input_reg); // Back up input_reg. |
3564 __ subsd(input_reg, xmm_scratch); | 3564 __ subsd(input_reg, xmm_scratch); |
3565 __ cvttsd2si(output_reg, input_reg); | 3565 __ cvttsd2si(output_reg, input_reg); |
3566 // Catch minint due to overflow, and to prevent overflow when compensating. | 3566 // Catch minint due to overflow, and to prevent overflow when compensating. |
3567 __ cmpl(output_reg, Immediate(0x80000000)); | 3567 __ cmpl(output_reg, Immediate(0x80000000)); |
3568 __ RecordComment("D2I conversion overflow"); | 3568 __ RecordComment("D2I conversion overflow"); |
3569 DeoptimizeIf(equal, instr->environment()); | 3569 DeoptimizeIf(equal, instr->environment()); |
3570 | 3570 |
3571 __ cvtlsi2sd(xmm_scratch, output_reg); | 3571 __ Cvtlsi2sd(xmm_scratch, output_reg); |
3572 __ ucomisd(input_reg, xmm_scratch); | 3572 __ ucomisd(input_reg, xmm_scratch); |
3573 __ j(equal, &restore, Label::kNear); | 3573 __ j(equal, &restore, Label::kNear); |
3574 __ subl(output_reg, Immediate(1)); | 3574 __ subl(output_reg, Immediate(1)); |
3575 // No overflow because we already ruled out minint. | 3575 // No overflow because we already ruled out minint. |
3576 __ bind(&restore); | 3576 __ bind(&restore); |
3577 __ movq(input_reg, kScratchRegister); // Restore input_reg. | 3577 __ movq(input_reg, kScratchRegister); // Restore input_reg. |
3578 __ jmp(&done); | 3578 __ jmp(&done); |
3579 | 3579 |
3580 __ bind(&round_to_zero); | 3580 __ bind(&round_to_zero); |
3581 // We return 0 for the input range [+0, 0.5[, or [-0.5, 0.5[ if | 3581 // We return 0 for the input range [+0, 0.5[, or [-0.5, 0.5[ if |
(...skipping 885 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4467 __ StoreToSafepointRegisterSlot(result, rax); | 4467 __ StoreToSafepointRegisterSlot(result, rax); |
4468 } | 4468 } |
4469 | 4469 |
4470 | 4470 |
4471 void LCodeGen::DoInteger32ToDouble(LInteger32ToDouble* instr) { | 4471 void LCodeGen::DoInteger32ToDouble(LInteger32ToDouble* instr) { |
4472 LOperand* input = instr->value(); | 4472 LOperand* input = instr->value(); |
4473 ASSERT(input->IsRegister() || input->IsStackSlot()); | 4473 ASSERT(input->IsRegister() || input->IsStackSlot()); |
4474 LOperand* output = instr->result(); | 4474 LOperand* output = instr->result(); |
4475 ASSERT(output->IsDoubleRegister()); | 4475 ASSERT(output->IsDoubleRegister()); |
4476 if (input->IsRegister()) { | 4476 if (input->IsRegister()) { |
4477 __ cvtlsi2sd(ToDoubleRegister(output), ToRegister(input)); | 4477 __ Cvtlsi2sd(ToDoubleRegister(output), ToRegister(input)); |
4478 } else { | 4478 } else { |
4479 __ cvtlsi2sd(ToDoubleRegister(output), ToOperand(input)); | 4479 __ Cvtlsi2sd(ToDoubleRegister(output), ToOperand(input)); |
4480 } | 4480 } |
4481 } | 4481 } |
4482 | 4482 |
4483 | 4483 |
4484 void LCodeGen::DoInteger32ToSmi(LInteger32ToSmi* instr) { | 4484 void LCodeGen::DoInteger32ToSmi(LInteger32ToSmi* instr) { |
4485 LOperand* input = instr->value(); | 4485 LOperand* input = instr->value(); |
4486 ASSERT(input->IsRegister()); | 4486 ASSERT(input->IsRegister()); |
4487 LOperand* output = instr->result(); | 4487 LOperand* output = instr->result(); |
4488 __ Integer32ToSmi(ToRegister(output), ToRegister(input)); | 4488 __ Integer32ToSmi(ToRegister(output), ToRegister(input)); |
4489 if (!instr->hydrogen()->value()->HasRange() || | 4489 if (!instr->hydrogen()->value()->HasRange() || |
(...skipping 196 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4686 DeoptimizeIf(not_zero, env); | 4686 DeoptimizeIf(not_zero, env); |
4687 } | 4687 } |
4688 __ jmp(&done, Label::kNear); | 4688 __ jmp(&done, Label::kNear); |
4689 } else { | 4689 } else { |
4690 ASSERT(mode == NUMBER_CANDIDATE_IS_SMI); | 4690 ASSERT(mode == NUMBER_CANDIDATE_IS_SMI); |
4691 } | 4691 } |
4692 | 4692 |
4693 // Smi to XMM conversion | 4693 // Smi to XMM conversion |
4694 __ bind(&load_smi); | 4694 __ bind(&load_smi); |
4695 __ SmiToInteger32(kScratchRegister, input_reg); | 4695 __ SmiToInteger32(kScratchRegister, input_reg); |
4696 __ cvtlsi2sd(result_reg, kScratchRegister); | 4696 __ Cvtlsi2sd(result_reg, kScratchRegister); |
4697 __ bind(&done); | 4697 __ bind(&done); |
4698 } | 4698 } |
4699 | 4699 |
4700 | 4700 |
4701 void LCodeGen::DoDeferredTaggedToI(LTaggedToI* instr) { | 4701 void LCodeGen::DoDeferredTaggedToI(LTaggedToI* instr) { |
4702 Label done, heap_number; | 4702 Label done, heap_number; |
4703 Register input_reg = ToRegister(instr->value()); | 4703 Register input_reg = ToRegister(instr->value()); |
4704 | 4704 |
4705 // Heap number map check. | 4705 // Heap number map check. |
4706 __ CompareRoot(FieldOperand(input_reg, HeapObject::kMapOffset), | 4706 __ CompareRoot(FieldOperand(input_reg, HeapObject::kMapOffset), |
(...skipping 15 matching lines...) Expand all Loading... |
4722 __ Set(kScratchRegister, V8_UINT64_C(0x8000000000000000)); | 4722 __ Set(kScratchRegister, V8_UINT64_C(0x8000000000000000)); |
4723 __ cmpq(input_reg, kScratchRegister); | 4723 __ cmpq(input_reg, kScratchRegister); |
4724 DeoptimizeIf(equal, instr->environment()); | 4724 DeoptimizeIf(equal, instr->environment()); |
4725 } else { | 4725 } else { |
4726 // Deoptimize if we don't have a heap number. | 4726 // Deoptimize if we don't have a heap number. |
4727 DeoptimizeIf(not_equal, instr->environment()); | 4727 DeoptimizeIf(not_equal, instr->environment()); |
4728 | 4728 |
4729 XMMRegister xmm_temp = ToDoubleRegister(instr->temp()); | 4729 XMMRegister xmm_temp = ToDoubleRegister(instr->temp()); |
4730 __ movsd(xmm0, FieldOperand(input_reg, HeapNumber::kValueOffset)); | 4730 __ movsd(xmm0, FieldOperand(input_reg, HeapNumber::kValueOffset)); |
4731 __ cvttsd2si(input_reg, xmm0); | 4731 __ cvttsd2si(input_reg, xmm0); |
4732 __ cvtlsi2sd(xmm_temp, input_reg); | 4732 __ Cvtlsi2sd(xmm_temp, input_reg); |
4733 __ ucomisd(xmm0, xmm_temp); | 4733 __ ucomisd(xmm0, xmm_temp); |
4734 DeoptimizeIf(not_equal, instr->environment()); | 4734 DeoptimizeIf(not_equal, instr->environment()); |
4735 DeoptimizeIf(parity_even, instr->environment()); // NaN. | 4735 DeoptimizeIf(parity_even, instr->environment()); // NaN. |
4736 if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { | 4736 if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { |
4737 __ testl(input_reg, input_reg); | 4737 __ testl(input_reg, input_reg); |
4738 __ j(not_zero, &done); | 4738 __ j(not_zero, &done); |
4739 __ movmskpd(input_reg, xmm0); | 4739 __ movmskpd(input_reg, xmm0); |
4740 __ andl(input_reg, Immediate(1)); | 4740 __ andl(input_reg, Immediate(1)); |
4741 DeoptimizeIf(not_zero, instr->environment()); | 4741 DeoptimizeIf(not_zero, instr->environment()); |
4742 } | 4742 } |
(...skipping 61 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4804 // Performs a truncating conversion of a floating point number as used by | 4804 // Performs a truncating conversion of a floating point number as used by |
4805 // the JS bitwise operations. | 4805 // the JS bitwise operations. |
4806 __ cvttsd2siq(result_reg, input_reg); | 4806 __ cvttsd2siq(result_reg, input_reg); |
4807 __ movq(kScratchRegister, | 4807 __ movq(kScratchRegister, |
4808 V8_INT64_C(0x8000000000000000), | 4808 V8_INT64_C(0x8000000000000000), |
4809 RelocInfo::NONE64); | 4809 RelocInfo::NONE64); |
4810 __ cmpq(result_reg, kScratchRegister); | 4810 __ cmpq(result_reg, kScratchRegister); |
4811 DeoptimizeIf(equal, instr->environment()); | 4811 DeoptimizeIf(equal, instr->environment()); |
4812 } else { | 4812 } else { |
4813 __ cvttsd2si(result_reg, input_reg); | 4813 __ cvttsd2si(result_reg, input_reg); |
4814 __ cvtlsi2sd(xmm0, result_reg); | 4814 __ Cvtlsi2sd(xmm0, result_reg); |
4815 __ ucomisd(xmm0, input_reg); | 4815 __ ucomisd(xmm0, input_reg); |
4816 DeoptimizeIf(not_equal, instr->environment()); | 4816 DeoptimizeIf(not_equal, instr->environment()); |
4817 DeoptimizeIf(parity_even, instr->environment()); // NaN. | 4817 DeoptimizeIf(parity_even, instr->environment()); // NaN. |
4818 if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { | 4818 if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { |
4819 Label done; | 4819 Label done; |
4820 // The integer converted back is equal to the original. We | 4820 // The integer converted back is equal to the original. We |
4821 // only have to test if we got -0 as an input. | 4821 // only have to test if we got -0 as an input. |
4822 __ testl(result_reg, result_reg); | 4822 __ testl(result_reg, result_reg); |
4823 __ j(not_zero, &done, Label::kNear); | 4823 __ j(not_zero, &done, Label::kNear); |
4824 __ movmskpd(result_reg, input_reg); | 4824 __ movmskpd(result_reg, input_reg); |
(...skipping 13 matching lines...) Expand all Loading... |
4838 ASSERT(input->IsDoubleRegister()); | 4838 ASSERT(input->IsDoubleRegister()); |
4839 LOperand* result = instr->result(); | 4839 LOperand* result = instr->result(); |
4840 ASSERT(result->IsRegister()); | 4840 ASSERT(result->IsRegister()); |
4841 CpuFeatureScope scope(masm(), SSE2); | 4841 CpuFeatureScope scope(masm(), SSE2); |
4842 | 4842 |
4843 XMMRegister input_reg = ToDoubleRegister(input); | 4843 XMMRegister input_reg = ToDoubleRegister(input); |
4844 Register result_reg = ToRegister(result); | 4844 Register result_reg = ToRegister(result); |
4845 | 4845 |
4846 Label done; | 4846 Label done; |
4847 __ cvttsd2si(result_reg, input_reg); | 4847 __ cvttsd2si(result_reg, input_reg); |
4848 __ cvtlsi2sd(xmm0, result_reg); | 4848 __ Cvtlsi2sd(xmm0, result_reg); |
4849 __ ucomisd(xmm0, input_reg); | 4849 __ ucomisd(xmm0, input_reg); |
4850 DeoptimizeIf(not_equal, instr->environment()); | 4850 DeoptimizeIf(not_equal, instr->environment()); |
4851 DeoptimizeIf(parity_even, instr->environment()); // NaN. | 4851 DeoptimizeIf(parity_even, instr->environment()); // NaN. |
4852 | 4852 |
4853 if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { | 4853 if (instr->hydrogen()->CheckFlag(HValue::kBailoutOnMinusZero)) { |
4854 // The integer converted back is equal to the original. We | 4854 // The integer converted back is equal to the original. We |
4855 // only have to test if we got -0 as an input. | 4855 // only have to test if we got -0 as an input. |
4856 __ testl(result_reg, result_reg); | 4856 __ testl(result_reg, result_reg); |
4857 __ j(not_zero, &done, Label::kNear); | 4857 __ j(not_zero, &done, Label::kNear); |
4858 __ movmskpd(result_reg, input_reg); | 4858 __ movmskpd(result_reg, input_reg); |
(...skipping 704 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
5563 FixedArray::kHeaderSize - kPointerSize)); | 5563 FixedArray::kHeaderSize - kPointerSize)); |
5564 __ bind(&done); | 5564 __ bind(&done); |
5565 } | 5565 } |
5566 | 5566 |
5567 | 5567 |
5568 #undef __ | 5568 #undef __ |
5569 | 5569 |
5570 } } // namespace v8::internal | 5570 } } // namespace v8::internal |
5571 | 5571 |
5572 #endif // V8_TARGET_ARCH_X64 | 5572 #endif // V8_TARGET_ARCH_X64 |
OLD | NEW |