OLD | NEW |
1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 the V8 project authors. All rights reserved. |
2 // Rrdistribution and use in source and binary forms, with or without | 2 // Rrdistribution 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 // * Rrdistributions of source code must retain the above copyright | 6 // * Rrdistributions 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 // * Rrdistributions in binary form must reproduce the above | 8 // * Rrdistributions 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 15 matching lines...) Expand all Loading... |
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
27 | 27 |
28 #include <stdlib.h> | 28 #include <stdlib.h> |
29 | 29 |
30 #include "src/v8.h" | 30 #include "src/v8.h" |
31 | 31 |
32 #include "src/base/platform/platform.h" | 32 #include "src/base/platform/platform.h" |
33 #include "src/code-stubs.h" | 33 #include "src/code-stubs.h" |
34 #include "src/factory.h" | 34 #include "src/factory.h" |
35 #include "src/macro-assembler.h" | 35 #include "src/macro-assembler.h" |
| 36 #include "src/register-configuration.h" |
36 #include "test/cctest/cctest.h" | 37 #include "test/cctest/cctest.h" |
37 #include "test/cctest/test-code-stubs.h" | 38 #include "test/cctest/test-code-stubs.h" |
38 | 39 |
39 using namespace v8::internal; | 40 using namespace v8::internal; |
40 | 41 |
41 | 42 |
42 #define __ assm. | 43 #define __ assm. |
43 | 44 |
44 ConvertDToIFunc MakeConvertDToIFuncTrampoline(Isolate* isolate, | 45 ConvertDToIFunc MakeConvertDToIFuncTrampoline(Isolate* isolate, |
45 Register source_reg, | 46 Register source_reg, |
46 Register destination_reg) { | 47 Register destination_reg) { |
47 // Allocate an executable page of memory. | 48 // Allocate an executable page of memory. |
48 size_t actual_size; | 49 size_t actual_size; |
49 byte* buffer = static_cast<byte*>(v8::base::OS::Allocate( | 50 byte* buffer = static_cast<byte*>(v8::base::OS::Allocate( |
50 Assembler::kMinimalBufferSize, &actual_size, true)); | 51 Assembler::kMinimalBufferSize, &actual_size, true)); |
51 CHECK(buffer); | 52 CHECK(buffer); |
52 HandleScope handles(isolate); | 53 HandleScope handles(isolate); |
53 MacroAssembler assm(isolate, buffer, static_cast<int>(actual_size)); | 54 MacroAssembler assm(isolate, buffer, static_cast<int>(actual_size)); |
54 int offset = | 55 int offset = |
55 source_reg.is(rsp) ? 0 : (HeapNumber::kValueOffset - kSmiTagSize); | 56 source_reg.is(rsp) ? 0 : (HeapNumber::kValueOffset - kSmiTagSize); |
56 DoubleToIStub stub(isolate, source_reg, destination_reg, offset, true); | 57 DoubleToIStub stub(isolate, source_reg, destination_reg, offset, true); |
57 byte* start = stub.GetCode()->instruction_start(); | 58 byte* start = stub.GetCode()->instruction_start(); |
58 | 59 |
59 __ pushq(rbx); | 60 __ pushq(rbx); |
60 __ pushq(rcx); | 61 __ pushq(rcx); |
61 __ pushq(rdx); | 62 __ pushq(rdx); |
62 __ pushq(rsi); | 63 __ pushq(rsi); |
63 __ pushq(rdi); | 64 __ pushq(rdi); |
64 | 65 |
| 66 |
| 67 const RegisterConfiguration* config = RegisterConfiguration::ArchDefault(); |
65 if (!source_reg.is(rsp)) { | 68 if (!source_reg.is(rsp)) { |
66 // The argument we pass to the stub is not a heap number, but instead | 69 // The argument we pass to the stub is not a heap number, but instead |
67 // stack-allocated and offset-wise made to look like a heap number for | 70 // stack-allocated and offset-wise made to look like a heap number for |
68 // the stub. We create that "heap number" after pushing all allocatable | 71 // the stub. We create that "heap number" after pushing all allocatable |
69 // registers. | 72 // registers. |
70 int double_argument_slot = | 73 int double_argument_slot = |
71 (Register::NumAllocatableRegisters() - 1) * kPointerSize + kDoubleSize; | 74 (config->num_allocatable_general_registers() - 1) * kPointerSize + |
| 75 kDoubleSize; |
72 __ leaq(source_reg, MemOperand(rsp, -double_argument_slot - offset)); | 76 __ leaq(source_reg, MemOperand(rsp, -double_argument_slot - offset)); |
73 } | 77 } |
74 | 78 |
75 // Save registers make sure they don't get clobbered. | 79 // Save registers make sure they don't get clobbered. |
76 int reg_num = 0; | 80 int reg_num = 0; |
77 for (;reg_num < Register::NumAllocatableRegisters(); ++reg_num) { | 81 for (; reg_num < config->num_allocatable_general_registers(); ++reg_num) { |
78 Register reg = Register::FromAllocationIndex(reg_num); | 82 Register reg = |
| 83 Register::from_code(config->GetAllocatableGeneralCode(reg_num)); |
79 if (!reg.is(rsp) && !reg.is(rbp) && !reg.is(destination_reg)) { | 84 if (!reg.is(rsp) && !reg.is(rbp) && !reg.is(destination_reg)) { |
80 __ pushq(reg); | 85 __ pushq(reg); |
81 } | 86 } |
82 } | 87 } |
83 | 88 |
84 // Put the double argument into the designated double argument slot. | 89 // Put the double argument into the designated double argument slot. |
85 __ subq(rsp, Immediate(kDoubleSize)); | 90 __ subq(rsp, Immediate(kDoubleSize)); |
86 __ movsd(MemOperand(rsp, 0), xmm0); | 91 __ movsd(MemOperand(rsp, 0), xmm0); |
87 | 92 |
88 // Call through to the actual stub | 93 // Call through to the actual stub |
89 __ Call(start, RelocInfo::EXTERNAL_REFERENCE); | 94 __ Call(start, RelocInfo::EXTERNAL_REFERENCE); |
90 | 95 |
91 __ addq(rsp, Immediate(kDoubleSize)); | 96 __ addq(rsp, Immediate(kDoubleSize)); |
92 | 97 |
93 // Make sure no registers have been unexpectedly clobbered | 98 // Make sure no registers have been unexpectedly clobbered |
94 for (--reg_num; reg_num >= 0; --reg_num) { | 99 for (--reg_num; reg_num >= 0; --reg_num) { |
95 Register reg = Register::FromAllocationIndex(reg_num); | 100 Register reg = |
| 101 Register::from_code(config->GetAllocatableGeneralCode(reg_num)); |
96 if (!reg.is(rsp) && !reg.is(rbp) && !reg.is(destination_reg)) { | 102 if (!reg.is(rsp) && !reg.is(rbp) && !reg.is(destination_reg)) { |
97 __ cmpq(reg, MemOperand(rsp, 0)); | 103 __ cmpq(reg, MemOperand(rsp, 0)); |
98 __ Assert(equal, kRegisterWasClobbered); | 104 __ Assert(equal, kRegisterWasClobbered); |
99 __ addq(rsp, Immediate(kPointerSize)); | 105 __ addq(rsp, Immediate(kPointerSize)); |
100 } | 106 } |
101 } | 107 } |
102 | 108 |
103 __ movq(rax, destination_reg); | 109 __ movq(rax, destination_reg); |
104 | 110 |
105 __ popq(rdi); | 111 __ popq(rdi); |
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
142 | 148 |
143 for (size_t s = 0; s < sizeof(source_registers) / sizeof(Register); s++) { | 149 for (size_t s = 0; s < sizeof(source_registers) / sizeof(Register); s++) { |
144 for (size_t d = 0; d < sizeof(dest_registers) / sizeof(Register); d++) { | 150 for (size_t d = 0; d < sizeof(dest_registers) / sizeof(Register); d++) { |
145 RunAllTruncationTests( | 151 RunAllTruncationTests( |
146 MakeConvertDToIFuncTrampoline(isolate, | 152 MakeConvertDToIFuncTrampoline(isolate, |
147 source_registers[s], | 153 source_registers[s], |
148 dest_registers[d])); | 154 dest_registers[d])); |
149 } | 155 } |
150 } | 156 } |
151 } | 157 } |
OLD | NEW |