]>
Commit | Line | Data |
---|---|---|
f9bf01c6 A |
1 | /* |
2 | * Copyright (C) 2009 University of Szeged | |
3 | * All rights reserved. | |
4 | * | |
5 | * Redistribution and use in source and binary forms, with or without | |
6 | * modification, are permitted provided that the following conditions | |
7 | * are met: | |
8 | * 1. Redistributions of source code must retain the above copyright | |
9 | * notice, this list of conditions and the following disclaimer. | |
10 | * 2. Redistributions in binary form must reproduce the above copyright | |
11 | * notice, this list of conditions and the following disclaimer in the | |
12 | * documentation and/or other materials provided with the distribution. | |
13 | * | |
14 | * THIS SOFTWARE IS PROVIDED BY UNIVERSITY OF SZEGED ``AS IS'' AND ANY | |
15 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
16 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | |
17 | * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL UNIVERSITY OF SZEGED OR | |
18 | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | |
19 | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | |
20 | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | |
21 | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | |
22 | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
23 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
24 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
25 | */ | |
26 | ||
27 | #include "config.h" | |
28 | ||
29 | #if ENABLE(ASSEMBLER) && CPU(ARM_TRADITIONAL) | |
30 | ||
31 | #include "ARMAssembler.h" | |
32 | ||
33 | namespace JSC { | |
34 | ||
35 | // Patching helpers | |
36 | ||
37 | void ARMAssembler::patchConstantPoolLoad(void* loadAddr, void* constPoolAddr) | |
38 | { | |
39 | ARMWord *ldr = reinterpret_cast<ARMWord*>(loadAddr); | |
40 | ARMWord diff = reinterpret_cast<ARMWord*>(constPoolAddr) - ldr; | |
41 | ARMWord index = (*ldr & 0xfff) >> 1; | |
42 | ||
43 | ASSERT(diff >= 1); | |
44 | if (diff >= 2 || index > 0) { | |
45 | diff = (diff + index - 2) * sizeof(ARMWord); | |
46 | ASSERT(diff <= 0xfff); | |
47 | *ldr = (*ldr & ~0xfff) | diff; | |
48 | } else | |
93a37866 | 49 | *ldr = (*ldr & ~(0xfff | ARMAssembler::DataTransferUp)) | sizeof(ARMWord); |
f9bf01c6 A |
50 | } |
51 | ||
52 | // Handle immediates | |
53 | ||
54 | ARMWord ARMAssembler::getOp2(ARMWord imm) | |
55 | { | |
56 | int rol; | |
57 | ||
58 | if (imm <= 0xff) | |
93a37866 | 59 | return Op2Immediate | imm; |
f9bf01c6 A |
60 | |
61 | if ((imm & 0xff000000) == 0) { | |
62 | imm <<= 8; | |
63 | rol = 8; | |
64 | } | |
65 | else { | |
66 | imm = (imm << 24) | (imm >> 8); | |
67 | rol = 0; | |
68 | } | |
69 | ||
70 | if ((imm & 0xff000000) == 0) { | |
71 | imm <<= 8; | |
72 | rol += 4; | |
73 | } | |
74 | ||
75 | if ((imm & 0xf0000000) == 0) { | |
76 | imm <<= 4; | |
77 | rol += 2; | |
78 | } | |
79 | ||
80 | if ((imm & 0xc0000000) == 0) { | |
81 | imm <<= 2; | |
82 | rol += 1; | |
83 | } | |
84 | ||
85 | if ((imm & 0x00ffffff) == 0) | |
93a37866 | 86 | return Op2Immediate | (imm >> 24) | (rol << 8); |
f9bf01c6 | 87 | |
93a37866 | 88 | return InvalidImmediate; |
f9bf01c6 A |
89 | } |
90 | ||
91 | int ARMAssembler::genInt(int reg, ARMWord imm, bool positive) | |
92 | { | |
93 | // Step1: Search a non-immediate part | |
94 | ARMWord mask; | |
95 | ARMWord imm1; | |
96 | ARMWord imm2; | |
97 | int rol; | |
98 | ||
99 | mask = 0xff000000; | |
100 | rol = 8; | |
101 | while(1) { | |
102 | if ((imm & mask) == 0) { | |
103 | imm = (imm << rol) | (imm >> (32 - rol)); | |
104 | rol = 4 + (rol >> 1); | |
105 | break; | |
106 | } | |
107 | rol += 2; | |
108 | mask >>= 2; | |
109 | if (mask & 0x3) { | |
110 | // rol 8 | |
111 | imm = (imm << 8) | (imm >> 24); | |
112 | mask = 0xff00; | |
113 | rol = 24; | |
114 | while (1) { | |
115 | if ((imm & mask) == 0) { | |
116 | imm = (imm << rol) | (imm >> (32 - rol)); | |
117 | rol = (rol >> 1) - 8; | |
118 | break; | |
119 | } | |
120 | rol += 2; | |
121 | mask >>= 2; | |
122 | if (mask & 0x3) | |
123 | return 0; | |
124 | } | |
125 | break; | |
126 | } | |
127 | } | |
128 | ||
129 | ASSERT((imm & 0xff) == 0); | |
130 | ||
131 | if ((imm & 0xff000000) == 0) { | |
93a37866 A |
132 | imm1 = Op2Immediate | ((imm >> 16) & 0xff) | (((rol + 4) & 0xf) << 8); |
133 | imm2 = Op2Immediate | ((imm >> 8) & 0xff) | (((rol + 8) & 0xf) << 8); | |
f9bf01c6 | 134 | } else if (imm & 0xc0000000) { |
93a37866 | 135 | imm1 = Op2Immediate | ((imm >> 24) & 0xff) | ((rol & 0xf) << 8); |
f9bf01c6 A |
136 | imm <<= 8; |
137 | rol += 4; | |
138 | ||
139 | if ((imm & 0xff000000) == 0) { | |
140 | imm <<= 8; | |
141 | rol += 4; | |
142 | } | |
143 | ||
144 | if ((imm & 0xf0000000) == 0) { | |
145 | imm <<= 4; | |
146 | rol += 2; | |
147 | } | |
148 | ||
149 | if ((imm & 0xc0000000) == 0) { | |
150 | imm <<= 2; | |
151 | rol += 1; | |
152 | } | |
153 | ||
154 | if ((imm & 0x00ffffff) == 0) | |
93a37866 | 155 | imm2 = Op2Immediate | (imm >> 24) | ((rol & 0xf) << 8); |
f9bf01c6 A |
156 | else |
157 | return 0; | |
158 | } else { | |
159 | if ((imm & 0xf0000000) == 0) { | |
160 | imm <<= 4; | |
161 | rol += 2; | |
162 | } | |
163 | ||
164 | if ((imm & 0xc0000000) == 0) { | |
165 | imm <<= 2; | |
166 | rol += 1; | |
167 | } | |
168 | ||
93a37866 | 169 | imm1 = Op2Immediate | ((imm >> 24) & 0xff) | ((rol & 0xf) << 8); |
f9bf01c6 A |
170 | imm <<= 8; |
171 | rol += 4; | |
172 | ||
173 | if ((imm & 0xf0000000) == 0) { | |
174 | imm <<= 4; | |
175 | rol += 2; | |
176 | } | |
177 | ||
178 | if ((imm & 0xc0000000) == 0) { | |
179 | imm <<= 2; | |
180 | rol += 1; | |
181 | } | |
182 | ||
183 | if ((imm & 0x00ffffff) == 0) | |
93a37866 | 184 | imm2 = Op2Immediate | (imm >> 24) | ((rol & 0xf) << 8); |
f9bf01c6 A |
185 | else |
186 | return 0; | |
187 | } | |
188 | ||
189 | if (positive) { | |
93a37866 A |
190 | mov(reg, imm1); |
191 | orr(reg, reg, imm2); | |
f9bf01c6 | 192 | } else { |
93a37866 A |
193 | mvn(reg, imm1); |
194 | bic(reg, reg, imm2); | |
f9bf01c6 A |
195 | } |
196 | ||
197 | return 1; | |
198 | } | |
199 | ||
200 | ARMWord ARMAssembler::getImm(ARMWord imm, int tmpReg, bool invert) | |
201 | { | |
202 | ARMWord tmp; | |
203 | ||
204 | // Do it by 1 instruction | |
205 | tmp = getOp2(imm); | |
93a37866 | 206 | if (tmp != InvalidImmediate) |
f9bf01c6 A |
207 | return tmp; |
208 | ||
209 | tmp = getOp2(~imm); | |
93a37866 | 210 | if (tmp != InvalidImmediate) { |
f9bf01c6 | 211 | if (invert) |
93a37866 A |
212 | return tmp | Op2InvertedImmediate; |
213 | mvn(tmpReg, tmp); | |
f9bf01c6 A |
214 | return tmpReg; |
215 | } | |
216 | ||
217 | return encodeComplexImm(imm, tmpReg); | |
218 | } | |
219 | ||
220 | void ARMAssembler::moveImm(ARMWord imm, int dest) | |
221 | { | |
222 | ARMWord tmp; | |
223 | ||
224 | // Do it by 1 instruction | |
225 | tmp = getOp2(imm); | |
93a37866 A |
226 | if (tmp != InvalidImmediate) { |
227 | mov(dest, tmp); | |
f9bf01c6 A |
228 | return; |
229 | } | |
230 | ||
231 | tmp = getOp2(~imm); | |
93a37866 A |
232 | if (tmp != InvalidImmediate) { |
233 | mvn(dest, tmp); | |
f9bf01c6 A |
234 | return; |
235 | } | |
236 | ||
237 | encodeComplexImm(imm, dest); | |
238 | } | |
239 | ||
240 | ARMWord ARMAssembler::encodeComplexImm(ARMWord imm, int dest) | |
241 | { | |
242 | #if WTF_ARM_ARCH_AT_LEAST(7) | |
243 | ARMWord tmp = getImm16Op2(imm); | |
93a37866 A |
244 | if (tmp != InvalidImmediate) { |
245 | movw(dest, tmp); | |
f9bf01c6 A |
246 | return dest; |
247 | } | |
93a37866 A |
248 | movw(dest, getImm16Op2(imm & 0xffff)); |
249 | movt(dest, getImm16Op2(imm >> 16)); | |
f9bf01c6 A |
250 | return dest; |
251 | #else | |
252 | // Do it by 2 instruction | |
253 | if (genInt(dest, imm, true)) | |
254 | return dest; | |
255 | if (genInt(dest, ~imm, false)) | |
256 | return dest; | |
257 | ||
93a37866 | 258 | ldrImmediate(dest, imm); |
f9bf01c6 A |
259 | return dest; |
260 | #endif | |
261 | } | |
262 | ||
263 | // Memory load/store helpers | |
264 | ||
93a37866 | 265 | void ARMAssembler::dataTransfer32(DataTransferTypeA transferType, RegisterID srcDst, RegisterID base, int32_t offset) |
f9bf01c6 A |
266 | { |
267 | if (offset >= 0) { | |
268 | if (offset <= 0xfff) | |
93a37866 | 269 | dtrUp(transferType, srcDst, base, offset); |
f9bf01c6 | 270 | else if (offset <= 0xfffff) { |
93a37866 A |
271 | add(ARMRegisters::S0, base, Op2Immediate | (offset >> 12) | (10 << 8)); |
272 | dtrUp(transferType, srcDst, ARMRegisters::S0, (offset & 0xfff)); | |
f9bf01c6 | 273 | } else { |
14957cd0 | 274 | moveImm(offset, ARMRegisters::S0); |
93a37866 | 275 | dtrUpRegister(transferType, srcDst, base, ARMRegisters::S0); |
f9bf01c6 A |
276 | } |
277 | } else { | |
6fe7ccc8 | 278 | if (offset >= -0xfff) |
93a37866 | 279 | dtrDown(transferType, srcDst, base, -offset); |
6fe7ccc8 | 280 | else if (offset >= -0xfffff) { |
93a37866 A |
281 | sub(ARMRegisters::S0, base, Op2Immediate | (-offset >> 12) | (10 << 8)); |
282 | dtrDown(transferType, srcDst, ARMRegisters::S0, (-offset & 0xfff)); | |
f9bf01c6 | 283 | } else { |
14957cd0 | 284 | moveImm(offset, ARMRegisters::S0); |
93a37866 | 285 | dtrUpRegister(transferType, srcDst, base, ARMRegisters::S0); |
f9bf01c6 A |
286 | } |
287 | } | |
288 | } | |
289 | ||
93a37866 | 290 | void ARMAssembler::baseIndexTransfer32(DataTransferTypeA transferType, RegisterID srcDst, RegisterID base, RegisterID index, int scale, int32_t offset) |
f9bf01c6 | 291 | { |
f9bf01c6 | 292 | ASSERT(scale >= 0 && scale <= 3); |
93a37866 | 293 | ARMWord op2 = lsl(index, scale); |
f9bf01c6 | 294 | |
93a37866 A |
295 | if (!offset) { |
296 | dtrUpRegister(transferType, srcDst, base, op2); | |
f9bf01c6 A |
297 | return; |
298 | } | |
93a37866 A |
299 | |
300 | if (offset <= 0xfffff && offset >= -0xfffff) { | |
301 | add(ARMRegisters::S0, base, op2); | |
302 | dataTransfer32(transferType, srcDst, ARMRegisters::S0, offset); | |
f9bf01c6 A |
303 | return; |
304 | } | |
305 | ||
93a37866 A |
306 | moveImm(offset, ARMRegisters::S0); |
307 | add(ARMRegisters::S0, ARMRegisters::S0, op2); | |
308 | dtrUpRegister(transferType, srcDst, base, ARMRegisters::S0); | |
309 | } | |
310 | ||
311 | void ARMAssembler::dataTransfer16(DataTransferTypeB transferType, RegisterID srcDst, RegisterID base, int32_t offset) | |
312 | { | |
313 | if (offset >= 0) { | |
314 | if (offset <= 0xff) | |
315 | halfDtrUp(transferType, srcDst, base, getOp2Half(offset)); | |
316 | else if (offset <= 0xffff) { | |
317 | add(ARMRegisters::S0, base, Op2Immediate | (offset >> 8) | (12 << 8)); | |
318 | halfDtrUp(transferType, srcDst, ARMRegisters::S0, getOp2Half(offset & 0xff)); | |
319 | } else { | |
320 | moveImm(offset, ARMRegisters::S0); | |
321 | halfDtrUpRegister(transferType, srcDst, base, ARMRegisters::S0); | |
322 | } | |
323 | } else { | |
324 | if (offset >= -0xff) | |
325 | halfDtrDown(transferType, srcDst, base, getOp2Half(-offset)); | |
326 | else if (offset >= -0xffff) { | |
327 | sub(ARMRegisters::S0, base, Op2Immediate | (-offset >> 8) | (12 << 8)); | |
328 | halfDtrDown(transferType, srcDst, ARMRegisters::S0, getOp2Half(-offset & 0xff)); | |
329 | } else { | |
330 | moveImm(offset, ARMRegisters::S0); | |
331 | halfDtrUpRegister(transferType, srcDst, base, ARMRegisters::S0); | |
332 | } | |
333 | } | |
f9bf01c6 A |
334 | } |
335 | ||
93a37866 A |
336 | void ARMAssembler::baseIndexTransfer16(DataTransferTypeB transferType, RegisterID srcDst, RegisterID base, RegisterID index, int scale, int32_t offset) |
337 | { | |
338 | if (!scale && !offset) { | |
339 | halfDtrUpRegister(transferType, srcDst, base, index); | |
340 | return; | |
341 | } | |
342 | ||
343 | ARMWord op2 = lsl(index, scale); | |
344 | ||
345 | if (offset <= 0xffff && offset >= -0xffff) { | |
346 | add(ARMRegisters::S0, base, op2); | |
347 | dataTransfer16(transferType, srcDst, ARMRegisters::S0, offset); | |
348 | return; | |
349 | } | |
350 | ||
351 | moveImm(offset, ARMRegisters::S0); | |
352 | add(ARMRegisters::S0, ARMRegisters::S0, op2); | |
353 | halfDtrUpRegister(transferType, srcDst, base, ARMRegisters::S0); | |
354 | } | |
355 | ||
356 | void ARMAssembler::dataTransferFloat(DataTransferTypeFloat transferType, FPRegisterID srcDst, RegisterID base, int32_t offset) | |
f9bf01c6 | 357 | { |
6fe7ccc8 A |
358 | // VFP cannot directly access memory that is not four-byte-aligned |
359 | if (!(offset & 0x3)) { | |
f9bf01c6 | 360 | if (offset <= 0x3ff && offset >= 0) { |
93a37866 | 361 | doubleDtrUp(transferType, srcDst, base, offset >> 2); |
f9bf01c6 A |
362 | return; |
363 | } | |
364 | if (offset <= 0x3ffff && offset >= 0) { | |
93a37866 A |
365 | add(ARMRegisters::S0, base, Op2Immediate | (offset >> 10) | (11 << 8)); |
366 | doubleDtrUp(transferType, srcDst, ARMRegisters::S0, (offset >> 2) & 0xff); | |
f9bf01c6 A |
367 | return; |
368 | } | |
369 | offset = -offset; | |
370 | ||
371 | if (offset <= 0x3ff && offset >= 0) { | |
93a37866 | 372 | doubleDtrDown(transferType, srcDst, base, offset >> 2); |
f9bf01c6 A |
373 | return; |
374 | } | |
375 | if (offset <= 0x3ffff && offset >= 0) { | |
93a37866 A |
376 | sub(ARMRegisters::S0, base, Op2Immediate | (offset >> 10) | (11 << 8)); |
377 | doubleDtrDown(transferType, srcDst, ARMRegisters::S0, (offset >> 2) & 0xff); | |
f9bf01c6 A |
378 | return; |
379 | } | |
380 | offset = -offset; | |
381 | } | |
382 | ||
93a37866 A |
383 | moveImm(offset, ARMRegisters::S0); |
384 | add(ARMRegisters::S0, ARMRegisters::S0, base); | |
385 | doubleDtrUp(transferType, srcDst, ARMRegisters::S0, 0); | |
386 | } | |
387 | ||
388 | void ARMAssembler::baseIndexTransferFloat(DataTransferTypeFloat transferType, FPRegisterID srcDst, RegisterID base, RegisterID index, int scale, int32_t offset) | |
389 | { | |
390 | add(ARMRegisters::S1, base, lsl(index, scale)); | |
391 | dataTransferFloat(transferType, srcDst, ARMRegisters::S1, offset); | |
f9bf01c6 A |
392 | } |
393 | ||
81345200 | 394 | void ARMAssembler::prepareExecutableCopy(void* to) |
f9bf01c6 A |
395 | { |
396 | // 64-bit alignment is required for next constant pool and JIT code as well | |
397 | m_buffer.flushWithoutBarrier(true); | |
14957cd0 | 398 | if (!m_buffer.isAligned(8)) |
f9bf01c6 A |
399 | bkpt(0); |
400 | ||
81345200 A |
401 | char* data = reinterpret_cast<char*>(m_buffer.data()); |
402 | ptrdiff_t delta = reinterpret_cast<char*>(to) - data; | |
f9bf01c6 A |
403 | |
404 | for (Jumps::Iterator iter = m_jumps.begin(); iter != m_jumps.end(); ++iter) { | |
405 | // The last bit is set if the constant must be placed on constant pool. | |
14957cd0 A |
406 | int pos = (iter->m_offset) & (~0x1); |
407 | ARMWord* ldrAddr = reinterpret_cast_ptr<ARMWord*>(data + pos); | |
f9bf01c6 | 408 | ARMWord* addr = getLdrImmAddress(ldrAddr); |
4e4e5a6f | 409 | if (*addr != InvalidBranchTarget) { |
14957cd0 | 410 | if (!(iter->m_offset & 1)) { |
93a37866 | 411 | intptr_t difference = reinterpret_cast_ptr<ARMWord*>(data + *addr) - (ldrAddr + DefaultPrefetchOffset); |
f9bf01c6 | 412 | |
93a37866 A |
413 | if ((difference <= MaximumBranchOffsetDistance && difference >= MinimumBranchOffsetDistance)) { |
414 | *ldrAddr = B | getConditionalField(*ldrAddr) | (difference & BranchOffsetMask); | |
f9bf01c6 A |
415 | continue; |
416 | } | |
417 | } | |
81345200 | 418 | *addr = reinterpret_cast<ARMWord>(data + delta + *addr); |
f9bf01c6 A |
419 | } |
420 | } | |
f9bf01c6 A |
421 | } |
422 | ||
423 | } // namespace JSC | |
424 | ||
425 | #endif // ENABLE(ASSEMBLER) && CPU(ARM_TRADITIONAL) |