Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(100)

Side by Side Diff: test/cctest/test-code-stubs-ia32.cc

Issue 18612005: Implement truncated d-to-i with a stub on x86 (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Add support for all register combinations Created 7 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright 2013 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are
4 // met:
5 //
6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided
11 // with the distribution.
12 // * Neither the name of Google Inc. nor the names of its
13 // contributors may be used to endorse or promote products derived
14 // from this software without specific prior written permission.
15 //
16 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28 #include <stdlib.h>
29
30 #include "v8.h"
31
32 #include "cctest.h"
33 #include "code-stubs.h"
34 #include "factory.h"
35 #include "macro-assembler.h"
36 #include "platform.h"
37
38 using namespace v8::internal;
39
40
41 typedef int32_t __stdcall (*ConvertDToIFunc)(double input);
42
43
44 int __stdcall ConvertDToICVersion(double d) {
45 Address double_ptr = reinterpret_cast<Address>(&d);
46 uint32_t exponent_bits = Memory::uint32_at(double_ptr + kDoubleSize / 2);
47 int32_t shifted_mask = static_cast<int32_t>(Double::kExponentMask >> 32);
48 int32_t exponent = (((exponent_bits & shifted_mask) >>
49 (Double::kPhysicalSignificandSize - 32)) -
50 HeapNumber::kExponentBias);
51 uint32_t unsigned_exponent = static_cast<uint32_t>(exponent);
52 int result = 0;
53 uint32_t max_exponent =
54 static_cast<uint32_t>(Double::kPhysicalSignificandSize);
55 if (unsigned_exponent >= max_exponent) {
56 if ((exponent - Double::kPhysicalSignificandSize) < 32) {
57 result = Memory::uint32_at(double_ptr) <<
58 (exponent - Double::kPhysicalSignificandSize);
59 }
60 } else {
61 uint64_t big_result =
62 (BitCast<uint64_t>(d) & Double::kSignificandMask) | Double::kHiddenBit;
63 big_result = big_result >> (Double::kPhysicalSignificandSize - exponent);
64 result = static_cast<uint32_t>(big_result);
65 }
66 if (static_cast<int32_t>(exponent_bits) < 0) {
67 return (0 - result);
68 } else {
69 return result;
70 }
71 }
72
73
74 void RunOneTruncationTestWithTest(ConvertDToIFunc func,
75 double from,
76 uint64_t to) {
77 int result = (*func)(from);
78 CHECK_EQ(to, static_cast<int>(result));
79 }
80
81
82 // #define NaN and Infinity so that it's possible to cut-and-paste these tests
83 // directly to a .js file and run them.
84 #define NaN NAN
85 #define Infinity INFINITY
86 #define RunOneTruncationTest(p1, p2) RunOneTruncationTestWithTest(func, p1, p2)
87
88 void RunAllTruncationTests(ConvertDToIFunc func) {
89 RunOneTruncationTest(0, 0);
90 RunOneTruncationTest(0.5, 0);
91 RunOneTruncationTest(-0.5, 0);
92 RunOneTruncationTest(1.5, 1);
93 RunOneTruncationTest(-1.5, -1);
94 RunOneTruncationTest(5.5, 5);
95 RunOneTruncationTest(-5.0, -5);
96 RunOneTruncationTest(NaN, 0);
97 RunOneTruncationTest(Infinity, 0);
98 RunOneTruncationTest(-NaN, 0);
99 RunOneTruncationTest(-Infinity, 0);
100
101 RunOneTruncationTest(4.5036e+15, 0x1635E000);
102 RunOneTruncationTest(-4.5036e+15, -372629504);
103
104 RunOneTruncationTest(4503603922337791.0, -1);
105 RunOneTruncationTest(-4503603922337791.0, 1);
106 RunOneTruncationTest(4503601774854143.0, 2147483647);
107 RunOneTruncationTest(-4503601774854143.0, -2147483647);
108 RunOneTruncationTest(9007207844675582.0, -2);
109 RunOneTruncationTest(-9007207844675582.0, 2);
110 RunOneTruncationTest(2.4178527921507624e+24, -536870912);
111 RunOneTruncationTest(-2.4178527921507624e+24, 536870912);
112 RunOneTruncationTest(2.417853945072267e+24, -536870912);
113 RunOneTruncationTest(-2.417853945072267e+24, 536870912);
114
115 RunOneTruncationTest(4.8357055843015248e+24, -1073741824);
116 RunOneTruncationTest(-4.8357055843015248e+24, 1073741824);
117 RunOneTruncationTest(4.8357078901445341e+24, -1073741824);
118 RunOneTruncationTest(-4.8357078901445341e+24, 1073741824);
119
120 RunOneTruncationTest(9.6714111686030497e+24, -2147483648);
121 RunOneTruncationTest(-9.6714111686030497e+24, -2147483648);
122 RunOneTruncationTest(9.6714157802890681e+24, -2147483648);
123 RunOneTruncationTest(-9.6714157802890681e+24, -2147483648);
124 }
125
126 #undef NaN
127 #undef Infinity
128 #undef RunOneTruncationTest
129
130 #define __ assm.
131
132 ConvertDToIFunc MakeConvertDToIFuncTrampoline(Isolate* isolate,
133 Register source_reg,
134 Register destination_reg) {
135 // Allocate an executable page of memory.
136 size_t actual_size;
137 byte* buffer = static_cast<byte*>(OS::Allocate(Assembler::kMinimalBufferSize,
138 &actual_size,
139 true));
140 CHECK(buffer);
141 HandleScope handles(isolate);
142 MacroAssembler assm(isolate, buffer, static_cast<int>(actual_size));
143 assm.set_allow_stub_calls(false);
144 int offset =
145 source_reg.is(esp) ? 0 : (HeapNumber::kValueOffset - kSmiTagSize);
146 DoubleToIStub stub(source_reg, destination_reg, offset, true);
147 byte* start = stub.GetCode(isolate)->instruction_start();
148
149 __ push(ebx);
150 __ push(ecx);
151 __ push(edx);
152 __ push(esi);
153 __ push(edi);
154
155 if (!source_reg.is(esp)) {
156 __ lea(source_reg, MemOperand(esp, 6 * kPointerSize - offset));
157 }
158
159 int param_offset = 7 * kPointerSize;
160 // Save registers make sure they don't get clobbered.
161 int reg_num = 0;
162 for (;reg_num < Register::NumAllocatableRegisters(); ++reg_num) {
163 Register reg = Register::from_code(reg_num);
164 if (!reg.is(esp) && !reg.is(ebp) && !reg.is(destination_reg)) {
165 __ push(reg);
166 param_offset += kPointerSize;
167 }
168 }
169
170 // Re-push the double argument
171 __ push(MemOperand(esp, param_offset));
172 __ push(MemOperand(esp, param_offset));
173
174 // Call through to the actual stub
175 __ call(start, RelocInfo::EXTERNAL_REFERENCE);
176 __ mov(eax, destination_reg);
177
178 __ add(esp, Immediate(kDoubleSize));
179
180 // Make sure no registers have been unexpectedly clobbered
181 for (--reg_num; reg_num >= 0; --reg_num) {
182 Register reg = Register::from_code(reg_num);
183 if (!reg.is(esp) && !reg.is(ebp) && !reg.is(destination_reg)) {
184 __ cmp(reg, MemOperand(esp, 0));
185 __ Assert(equal, "register was clobbered");
186 __ add(esp, Immediate(kPointerSize));
187 }
188 }
189
190 __ pop(edi);
191 __ pop(esi);
192 __ pop(edx);
193 __ pop(ecx);
194 __ pop(ebx);
195
196 __ ret(kDoubleSize);
197
198 CodeDesc desc;
199 assm.GetCode(&desc);
200 return reinterpret_cast<ConvertDToIFunc>(buffer);
201 }
202
203 #undef __
204
205
206 static Isolate* GetIsolateFrom(LocalContext* context) {
207 return reinterpret_cast<Isolate*>((*context)->GetIsolate());
208 }
209
210
211 TEST(ConvertDToI) {
212 CcTest::InitializeVM();
213 LocalContext context;
214 Isolate* isolate = GetIsolateFrom(&context);
215 HandleScope scope(isolate);
216
217 #if DEBUG
218 // Verify that the tests actually work with the C version. In the release
219 // code, the compiler optimizes it away because it's all constant, but does it
220 // wrong, triggering an assert on gcc.
221 RunAllTruncationTests(&ConvertDToICVersion);
222 #endif
223
224 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esp, eax));
225 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esp, ebx));
226 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esp, ecx));
227 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esp, edx));
228 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esp, edi));
229 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esp, esi));
230 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, eax, eax));
231 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, eax, ebx));
232 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, eax, ecx));
233 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, eax, edx));
234 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, eax, edi));
235 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, eax, esi));
236 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ebx, eax));
237 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ebx, ebx));
238 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ebx, ecx));
239 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ebx, edx));
240 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ebx, edi));
241 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ebx, esi));
242 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ecx, eax));
243 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ecx, ebx));
244 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ecx, ecx));
245 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ecx, edx));
246 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ecx, edi));
247 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, ecx, esi));
248 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edx, eax));
249 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edx, ebx));
250 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edx, ecx));
251 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edx, edx));
252 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edx, edi));
253 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edx, esi));
254 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esi, eax));
255 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esi, ebx));
256 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esi, ecx));
257 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esi, edx));
258 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esi, edi));
259 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, esi, esi));
260 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edi, eax));
261 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edi, ebx));
262 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edi, ecx));
263 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edi, edx));
264 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edi, edi));
265 RunAllTruncationTests(MakeConvertDToIFuncTrampoline(isolate, edi, esi));
266 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698