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 3649 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3660 __ CallStub(&stub); | 3660 __ CallStub(&stub); |
3661 } else { | 3661 } else { |
3662 ASSERT(exponent_type.IsDouble()); | 3662 ASSERT(exponent_type.IsDouble()); |
3663 MathPowStub stub(MathPowStub::DOUBLE); | 3663 MathPowStub stub(MathPowStub::DOUBLE); |
3664 __ CallStub(&stub); | 3664 __ CallStub(&stub); |
3665 } | 3665 } |
3666 } | 3666 } |
3667 | 3667 |
3668 | 3668 |
3669 void LCodeGen::DoRandom(LRandom* instr) { | 3669 void LCodeGen::DoRandom(LRandom* instr) { |
3670 class DeferredDoRandom V8_FINAL : public LDeferredCode { | 3670 // Assert that register size is twice the size of each seed. |
3671 public: | |
3672 DeferredDoRandom(LCodeGen* codegen, LRandom* instr) | |
3673 : LDeferredCode(codegen), instr_(instr) { } | |
3674 virtual void Generate() V8_OVERRIDE { codegen()->DoDeferredRandom(instr_); } | |
3675 virtual LInstruction* instr() V8_OVERRIDE { return instr_; } | |
3676 private: | |
3677 LRandom* instr_; | |
3678 }; | |
3679 | |
3680 DeferredDoRandom* deferred = new(zone()) DeferredDoRandom(this, instr); | |
3681 | |
3682 // Having marked this instruction as a call we can use any | |
3683 // registers. | |
3684 ASSERT(ToDoubleRegister(instr->result()).is(xmm1)); | |
3685 | |
3686 // Choose the right register for the first argument depending on | |
3687 // calling convention. | |
3688 #ifdef _WIN64 | |
3689 ASSERT(ToRegister(instr->global_object()).is(rcx)); | |
3690 Register global_object = rcx; | |
3691 #else | |
3692 ASSERT(ToRegister(instr->global_object()).is(rdi)); | |
3693 Register global_object = rdi; | |
3694 #endif | |
3695 | |
3696 static const int kSeedSize = sizeof(uint32_t); | 3671 static const int kSeedSize = sizeof(uint32_t); |
3697 STATIC_ASSERT(kPointerSize == 2 * kSeedSize); | 3672 STATIC_ASSERT(kPointerSize == 2 * kSeedSize); |
3698 | 3673 |
3699 __ movq(global_object, | 3674 // Load native context |
3700 FieldOperand(global_object, GlobalObject::kNativeContextOffset)); | 3675 Register global_object = ToRegister(instr->global_object()); |
| 3676 Register native_context = global_object; |
| 3677 __ movq(native_context, FieldOperand( |
| 3678 global_object, GlobalObject::kNativeContextOffset)); |
| 3679 |
| 3680 // Load state (FixedArray of the native context's random seeds) |
3701 static const int kRandomSeedOffset = | 3681 static const int kRandomSeedOffset = |
3702 FixedArray::kHeaderSize + Context::RANDOM_SEED_INDEX * kPointerSize; | 3682 FixedArray::kHeaderSize + Context::RANDOM_SEED_INDEX * kPointerSize; |
3703 __ movq(rbx, FieldOperand(global_object, kRandomSeedOffset)); | 3683 Register state = native_context; |
3704 // rbx: FixedArray of the native context's random seeds | 3684 __ movq(state, FieldOperand(native_context, kRandomSeedOffset)); |
3705 | 3685 |
3706 // Load state[0]. | 3686 // Load state[0]. |
3707 __ movl(rax, FieldOperand(rbx, ByteArray::kHeaderSize)); | 3687 Register state0 = ToRegister(instr->scratch()); |
3708 // If state[0] == 0, call runtime to initialize seeds. | 3688 __ movl(state0, FieldOperand(state, ByteArray::kHeaderSize)); |
3709 __ testl(rax, rax); | |
3710 __ j(zero, deferred->entry()); | |
3711 // Load state[1]. | 3689 // Load state[1]. |
3712 __ movl(rcx, FieldOperand(rbx, ByteArray::kHeaderSize + kSeedSize)); | 3690 Register state1 = ToRegister(instr->scratch2()); |
| 3691 __ movl(state1, FieldOperand(state, ByteArray::kHeaderSize + kSeedSize)); |
3713 | 3692 |
3714 // state[0] = 18273 * (state[0] & 0xFFFF) + (state[0] >> 16) | 3693 // state[0] = 18273 * (state[0] & 0xFFFF) + (state[0] >> 16) |
3715 // Only operate on the lower 32 bit of rax. | 3694 Register scratch3 = ToRegister(instr->scratch3()); |
3716 __ movzxwl(rdx, rax); | 3695 __ movzxwl(scratch3, state0); |
3717 __ imull(rdx, rdx, Immediate(18273)); | 3696 __ imull(scratch3, scratch3, Immediate(18273)); |
3718 __ shrl(rax, Immediate(16)); | 3697 __ shrl(state0, Immediate(16)); |
3719 __ addl(rax, rdx); | 3698 __ addl(state0, scratch3); |
3720 // Save state[0]. | 3699 // Save state[0]. |
3721 __ movl(FieldOperand(rbx, ByteArray::kHeaderSize), rax); | 3700 __ movl(FieldOperand(state, ByteArray::kHeaderSize), state0); |
3722 | 3701 |
3723 // state[1] = 36969 * (state[1] & 0xFFFF) + (state[1] >> 16) | 3702 // state[1] = 36969 * (state[1] & 0xFFFF) + (state[1] >> 16) |
3724 __ movzxwl(rdx, rcx); | 3703 __ movzxwl(scratch3, state1); |
3725 __ imull(rdx, rdx, Immediate(36969)); | 3704 __ imull(scratch3, scratch3, Immediate(36969)); |
3726 __ shrl(rcx, Immediate(16)); | 3705 __ shrl(state1, Immediate(16)); |
3727 __ addl(rcx, rdx); | 3706 __ addl(state1, scratch3); |
3728 // Save state[1]. | 3707 // Save state[1]. |
3729 __ movl(FieldOperand(rbx, ByteArray::kHeaderSize + kSeedSize), rcx); | 3708 __ movl(FieldOperand(state, ByteArray::kHeaderSize + kSeedSize), state1); |
3730 | 3709 |
3731 // Random bit pattern = (state[0] << 14) + (state[1] & 0x3FFFF) | 3710 // Random bit pattern = (state[0] << 14) + (state[1] & 0x3FFFF) |
3732 __ shll(rax, Immediate(14)); | 3711 Register random = state0; |
3733 __ andl(rcx, Immediate(0x3FFFF)); | 3712 __ shll(random, Immediate(14)); |
3734 __ addl(rax, rcx); | 3713 __ andl(state1, Immediate(0x3FFFF)); |
| 3714 __ addl(random, state1); |
3735 | 3715 |
3736 __ bind(deferred->exit()); | |
3737 // Convert 32 random bits in rax to 0.(32 random bits) in a double | 3716 // Convert 32 random bits in rax to 0.(32 random bits) in a double |
3738 // by computing: | 3717 // by computing: |
3739 // ( 1.(20 0s)(32 random bits) x 2^20 ) - (1.0 x 2^20)). | 3718 // ( 1.(20 0s)(32 random bits) x 2^20 ) - (1.0 x 2^20)). |
3740 __ movq(rcx, V8_INT64_C(0x4130000000000000), | 3719 XMMRegister result = ToDoubleRegister(instr->result()); |
| 3720 // We use xmm0 as fixed scratch register here. |
| 3721 XMMRegister scratch4 = xmm0; |
| 3722 __ movq(scratch3, V8_INT64_C(0x4130000000000000), |
3741 RelocInfo::NONE64); // 1.0 x 2^20 as double | 3723 RelocInfo::NONE64); // 1.0 x 2^20 as double |
3742 __ movq(xmm2, rcx); | 3724 __ movq(scratch4, scratch3); |
3743 __ movd(xmm1, rax); | 3725 __ movd(result, random); |
3744 __ xorps(xmm1, xmm2); | 3726 __ xorps(result, scratch4); |
3745 __ subsd(xmm1, xmm2); | 3727 __ subsd(result, scratch4); |
3746 } | 3728 } |
3747 | 3729 |
3748 | 3730 |
3749 void LCodeGen::DoDeferredRandom(LRandom* instr) { | |
3750 __ PrepareCallCFunction(1); | |
3751 __ CallCFunction(ExternalReference::random_uint32_function(isolate()), 1); | |
3752 __ movq(rsi, Operand(rbp, StandardFrameConstants::kContextOffset)); | |
3753 // Return value is in rax. | |
3754 } | |
3755 | |
3756 | |
3757 void LCodeGen::DoMathExp(LMathExp* instr) { | 3731 void LCodeGen::DoMathExp(LMathExp* instr) { |
3758 XMMRegister input = ToDoubleRegister(instr->value()); | 3732 XMMRegister input = ToDoubleRegister(instr->value()); |
3759 XMMRegister result = ToDoubleRegister(instr->result()); | 3733 XMMRegister result = ToDoubleRegister(instr->result()); |
3760 Register temp1 = ToRegister(instr->temp1()); | 3734 Register temp1 = ToRegister(instr->temp1()); |
3761 Register temp2 = ToRegister(instr->temp2()); | 3735 Register temp2 = ToRegister(instr->temp2()); |
3762 | 3736 |
3763 MathExpGenerator::EmitMathExp(masm(), input, result, xmm0, temp1, temp2); | 3737 MathExpGenerator::EmitMathExp(masm(), input, result, xmm0, temp1, temp2); |
3764 } | 3738 } |
3765 | 3739 |
3766 | 3740 |
(...skipping 1752 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
5519 FixedArray::kHeaderSize - kPointerSize)); | 5493 FixedArray::kHeaderSize - kPointerSize)); |
5520 __ bind(&done); | 5494 __ bind(&done); |
5521 } | 5495 } |
5522 | 5496 |
5523 | 5497 |
5524 #undef __ | 5498 #undef __ |
5525 | 5499 |
5526 } } // namespace v8::internal | 5500 } } // namespace v8::internal |
5527 | 5501 |
5528 #endif // V8_TARGET_ARCH_X64 | 5502 #endif // V8_TARGET_ARCH_X64 |
OLD | NEW |