]> git.saurik.com Git - apple/javascriptcore.git/blob - offlineasm/cloop.rb
JavaScriptCore-7600.1.4.17.5.tar.gz
[apple/javascriptcore.git] / offlineasm / cloop.rb
1 # Copyright (C) 2012, 2014 Apple Inc. All rights reserved.
2 #
3 # Redistribution and use in source and binary forms, with or without
4 # modification, are permitted provided that the following conditions
5 # are met:
6 # 1. Redistributions of source code must retain the above copyright
7 # notice, this list of conditions and the following disclaimer.
8 # 2. Redistributions in binary form must reproduce the above copyright
9 # notice, this list of conditions and the following disclaimer in the
10 # documentation and/or other materials provided with the distribution.
11 #
12 # THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
13 # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
14 # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
15 # PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
16 # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
17 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
18 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
19 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
20 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
21 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
22 # THE POSSIBILITY OF SUCH DAMAGE.
23
24 require "config"
25 require "ast"
26 require "opt"
27
28 # The CLoop llint backend is initially based on the ARMv7 backend, and
29 # then further enhanced with a few instructions from the x86 backend to
30 # support building for X64 targets. Hence, the shape of the generated
31 # code and the usage convention of registers will look a lot like the
32 # ARMv7 backend's.
33
34 def cloopMapType(type)
35 case type
36 when :int; ".i"
37 when :uint; ".u"
38 when :int32; ".i32"
39 when :uint32; ".u32"
40 when :int64; ".i64"
41 when :uint64; ".u64"
42 when :int8; ".i8"
43 when :uint8; ".u8"
44 when :int8Ptr; ".i8p"
45 when :voidPtr; ".vp"
46 when :nativeFunc; ".nativeFunc"
47 when :double; ".d"
48 when :castToDouble; ".castToDouble"
49 when :castToInt64; ".castToInt64"
50 when :opcode; ".opcode"
51 else;
52 raise "Unsupported type"
53 end
54 end
55
56
57 class SpecialRegister < NoChildren
58 def clDump
59 @name
60 end
61 def clValue(type=:int)
62 @name + cloopMapType(type)
63 end
64 end
65
66 C_LOOP_SCRATCH_FPR = SpecialRegister.new("d6")
67
68 class RegisterID
69 def clDump
70 case name
71 # The cloop is modelled on the ARM implementation. Hence, the a0-a3
72 # registers are aliases for r0-r3 i.e. t0-t3 in our case.
73 when "t0", "a0"
74 "t0"
75 when "t1", "a1"
76 "t1"
77 when "t2", "a2"
78 "t2"
79 when "t3", "a3"
80 "t3"
81 when "t4"
82 "pc"
83 when "t5"
84 "t5"
85 when "t6"
86 "pcBase"
87 when "t7"
88 "t7"
89 when "csr1"
90 "tagTypeNumber"
91 when "csr2"
92 "tagMask"
93 when "cfr"
94 "cfr"
95 when "lr"
96 "lr"
97 when "sp"
98 "sp"
99 else
100 raise "Bad register #{name} for C_LOOP at #{codeOriginString}"
101 end
102 end
103 def clValue(type=:int)
104 clDump + cloopMapType(type)
105 end
106 end
107
108 class FPRegisterID
109 def clDump
110 case name
111 when "ft0", "fr"
112 "d0"
113 when "ft1"
114 "d1"
115 when "ft2"
116 "d2"
117 when "ft3"
118 "d3"
119 when "ft4"
120 "d4"
121 when "ft5"
122 "d5"
123 else
124 raise "Bad register #{name} for C_LOOP at #{codeOriginString}"
125 end
126 end
127 def clValue(type=:int)
128 clDump + cloopMapType(type)
129 end
130 end
131
132 class Immediate
133 def clDump
134 "#{value}"
135 end
136 def clValue(type=:int)
137 # There is a case of a very large unsigned number (0x8000000000000000)
138 # which we wish to encode. Unfortunately, the C/C++ compiler
139 # complains if we express that number as a positive decimal integer.
140 # Hence, for positive values, we just convert the number into hex form
141 # to keep the compiler happy.
142 #
143 # However, for negative values, the to_s(16) hex conversion method does
144 # not strip the "-" sign resulting in a meaningless "0x-..." valueStr.
145 # To workaround this, we simply don't encode negative numbers as hex.
146
147 valueStr = (value < 0) ? "#{value}" : "0x#{value.to_s(16)}"
148
149 case type
150 when :int8; "int8_t(#{valueStr})"
151 when :int32; "int32_t(#{valueStr})"
152 when :int64; "int64_t(#{valueStr})"
153 when :int; "intptr_t(#{valueStr})"
154 when :uint8; "uint8_t(#{valueStr})"
155 when :uint32; "uint32_t(#{valueStr})"
156 when :uint64; "uint64_t(#{valueStr})"
157 when :uint; "uintptr_t(#{valueStr})"
158 else
159 raise "Not implemented immediate of type: #{type}"
160 end
161 end
162 end
163
164 class Address
165 def clDump
166 "[#{base.clDump}, #{offset.value}]"
167 end
168 def clValue(type=:int)
169 case type
170 when :int8; int8MemRef
171 when :int32; int32MemRef
172 when :int64; int64MemRef
173 when :int; intMemRef
174 when :uint8; uint8MemRef
175 when :uint32; uint32MemRef
176 when :uint64; uint64MemRef
177 when :uint; uintMemRef
178 when :opcode; opcodeMemRef
179 when :nativeFunc; nativeFuncMemRef
180 else
181 raise "Unexpected Address type: #{type}"
182 end
183 end
184 def pointerExpr
185 if offset.value == 0
186 "#{base.clValue(:int8Ptr)}"
187 elsif offset.value > 0
188 "#{base.clValue(:int8Ptr)} + #{offset.value}"
189 else
190 "#{base.clValue(:int8Ptr)} - #{-offset.value}"
191 end
192 end
193 def int8MemRef
194 "*CAST<int8_t*>(#{pointerExpr})"
195 end
196 def int16MemRef
197 "*CAST<int16_t*>(#{pointerExpr})"
198 end
199 def int32MemRef
200 "*CAST<int32_t*>(#{pointerExpr})"
201 end
202 def int64MemRef
203 "*CAST<int64_t*>(#{pointerExpr})"
204 end
205 def intMemRef
206 "*CAST<intptr_t*>(#{pointerExpr})"
207 end
208 def uint8MemRef
209 "*CAST<uint8_t*>(#{pointerExpr})"
210 end
211 def uint16MemRef
212 "*CAST<uint16_t*>(#{pointerExpr})"
213 end
214 def uint32MemRef
215 "*CAST<uint32_t*>(#{pointerExpr})"
216 end
217 def uint64MemRef
218 "*CAST<uint64_t*>(#{pointerExpr})"
219 end
220 def uintMemRef
221 "*CAST<uintptr_t*>(#{pointerExpr})"
222 end
223 def nativeFuncMemRef
224 "*CAST<NativeFunction*>(#{pointerExpr})"
225 end
226 def opcodeMemRef
227 "*CAST<Opcode*>(#{pointerExpr})"
228 end
229 def dblMemRef
230 "*CAST<double*>(#{pointerExpr})"
231 end
232 end
233
234 class BaseIndex
235 def clDump
236 "[#{base.clDump}, #{offset.clDump}, #{index.clDump} << #{scaleShift}]"
237 end
238 def clValue(type=:int)
239 case type
240 when :int8; int8MemRef
241 when :int32; int32MemRef
242 when :int64; int64MemRef
243 when :int; intMemRef
244 when :uint8; uint8MemRef
245 when :uint32; uint32MemRef
246 when :uint64; uint64MemRef
247 when :uint; uintMemRef
248 when :opcode; opcodeMemRef
249 else
250 raise "Unexpected BaseIndex type: #{type}"
251 end
252 end
253 def pointerExpr
254 if offset.value == 0
255 "#{base.clValue(:int8Ptr)} + (#{index.clValue} << #{scaleShift})"
256 else
257 "#{base.clValue(:int8Ptr)} + (#{index.clValue} << #{scaleShift}) + #{offset.clValue}"
258 end
259 end
260 def int8MemRef
261 "*CAST<int8_t*>(#{pointerExpr})"
262 end
263 def int16MemRef
264 "*CAST<int16_t*>(#{pointerExpr})"
265 end
266 def int32MemRef
267 "*CAST<int32_t*>(#{pointerExpr})"
268 end
269 def int64MemRef
270 "*CAST<int64_t*>(#{pointerExpr})"
271 end
272 def intMemRef
273 "*CAST<intptr_t*>(#{pointerExpr})"
274 end
275 def uint8MemRef
276 "*CAST<uint8_t*>(#{pointerExpr})"
277 end
278 def uint16MemRef
279 "*CAST<uint16_t*>(#{pointerExpr})"
280 end
281 def uint32MemRef
282 "*CAST<uint32_t*>(#{pointerExpr})"
283 end
284 def uint64MemRef
285 "*CAST<uint64_t*>(#{pointerExpr})"
286 end
287 def uintMemRef
288 "*CAST<uintptr_t*>(#{pointerExpr})"
289 end
290 def opcodeMemRef
291 "*CAST<Opcode*>(#{pointerExpr})"
292 end
293 def dblMemRef
294 "*CAST<double*>(#{pointerExpr})"
295 end
296 end
297
298 class AbsoluteAddress
299 def clDump
300 "#{codeOriginString}"
301 end
302 def clValue
303 clDump
304 end
305 end
306
307
308 #
309 # Lea support.
310 #
311
312 class Address
313 def cloopEmitLea(destination, type)
314 if destination == base
315 $asm.putc "#{destination.clValue(:int8Ptr)} += #{offset.clValue(type)};"
316 else
317 $asm.putc "#{destination.clValue(:int8Ptr)} = #{base.clValue(:int8Ptr)} + #{offset.clValue(type)};"
318 end
319 end
320 end
321
322 class BaseIndex
323 def cloopEmitLea(destination, type)
324 raise "Malformed BaseIndex, offset should be zero at #{codeOriginString}" unless offset.value == 0
325 $asm.putc "#{destination.clValue(:int8Ptr)} = #{base.clValue(:int8Ptr)} + (#{index.clValue} << #{scaleShift});"
326 end
327 end
328
329 #
330 # Actual lowering code follows.
331 #
332
333 class Sequence
334 def getModifiedListC_LOOP
335 myList = @list
336
337 # Verify that we will only see instructions and labels.
338 myList.each {
339 | node |
340 unless node.is_a? Instruction or
341 node.is_a? Label or
342 node.is_a? LocalLabel or
343 node.is_a? Skip
344 raise "Unexpected #{node.inspect} at #{node.codeOrigin}"
345 end
346 }
347
348 return myList
349 end
350 end
351
352 def clOperands(operands)
353 operands.map{|v| v.clDump}.join(", ")
354 end
355
356
357 def cloopEmitOperation(operands, type, operator)
358 raise unless type == :int || type == :uint || type == :int32 || type == :uint32 || \
359 type == :int64 || type == :uint64 || type == :double
360 if operands.size == 3
361 $asm.putc "#{operands[2].clValue(type)} = #{operands[0].clValue(type)} #{operator} #{operands[1].clValue(type)};"
362 if operands[2].is_a? RegisterID and (type == :int32 or type == :uint32)
363 $asm.putc "#{operands[2].clDump}.clearHighWord();" # Just clear it. It does nothing on the 32-bit port.
364 end
365 else
366 raise unless operands.size == 2
367 raise unless not operands[1].is_a? Immediate
368 $asm.putc "#{operands[1].clValue(type)} = #{operands[1].clValue(type)} #{operator} #{operands[0].clValue(type)};"
369 if operands[1].is_a? RegisterID and (type == :int32 or type == :uint32)
370 $asm.putc "#{operands[1].clDump}.clearHighWord();" # Just clear it. It does nothing on the 32-bit port.
371 end
372 end
373 end
374
375 def cloopEmitShiftOperation(operands, type, operator)
376 raise unless type == :int || type == :uint || type == :int32 || type == :uint32 || type == :int64 || type == :uint64
377 if operands.size == 3
378 $asm.putc "#{operands[2].clValue(type)} = #{operands[1].clValue(type)} #{operator} (#{operands[0].clValue(:int)} & 0x1f);"
379 if operands[2].is_a? RegisterID and (type == :int32 or type == :uint32)
380 $asm.putc "#{operands[2].clDump}.clearHighWord();" # Just clear it. It does nothing on the 32-bit port.
381 end
382 else
383 raise unless operands.size == 2
384 raise unless not operands[1].is_a? Immediate
385 $asm.putc "#{operands[1].clValue(type)} = #{operands[1].clValue(type)} #{operator} (#{operands[0].clValue(:int)} & 0x1f);"
386 if operands[1].is_a? RegisterID and (type == :int32 or type == :uint32)
387 $asm.putc "#{operands[1].clDump}.clearHighWord();" # Just clear it. It does nothing on the 32-bit port.
388 end
389 end
390 end
391
392 def cloopEmitUnaryOperation(operands, type, operator)
393 raise unless type == :int || type == :uint || type == :int32 || type == :uint32 || type == :int64 || type == :uint64
394 raise unless operands.size == 1
395 raise unless not operands[0].is_a? Immediate
396 $asm.putc "#{operands[0].clValue(type)} = #{operator}#{operands[0].clValue(type)};"
397 if operands[0].is_a? RegisterID and (type == :int32 or type == :uint32)
398 $asm.putc "#{operands[0].clDump}.clearHighWord();" # Just clear it. It does nothing on the 32-bit port.
399 end
400 end
401
402 def cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, condition)
403 $asm.putc "if (std::isnan(#{operands[0].clValue(:double)}) || std::isnan(#{operands[1].clValue(:double)})"
404 $asm.putc " || (#{operands[0].clValue(:double)} #{condition} #{operands[1].clValue(:double)}))"
405 $asm.putc " goto #{operands[2].cLabel};"
406 end
407
408
409 def cloopEmitCompareAndSet(operands, type, comparator)
410 # The result is a boolean. Hence, it doesn't need to be based on the type
411 # of the arguments being compared.
412 $asm.putc "#{operands[2].clValue} = (#{operands[0].clValue(type)} #{comparator} #{op2 = operands[1].clValue(type)});"
413 end
414
415
416 def cloopEmitCompareAndBranch(operands, type, comparator)
417 $asm.putc "if (#{operands[0].clValue(type)} #{comparator} #{operands[1].clValue(type)})"
418 $asm.putc " goto #{operands[2].cLabel};"
419 end
420
421
422 # conditionTest should contain a string that provides a comparator and a RHS
423 # value e.g. "< 0".
424 def cloopGenerateConditionExpression(operands, type, conditionTest)
425 op1 = operands[0].clValue(type)
426
427 # The operands must consist of 2 or 3 values.
428 case operands.size
429 when 2 # Just test op1 against the conditionTest.
430 lhs = op1
431 when 3 # Mask op1 with op2 before testing against the conditionTest.
432 lhs = "(#{op1} & #{operands[1].clValue(type)})"
433 else
434 raise "Expected 2 or 3 operands but got #{operands.size} at #{codeOriginString}"
435 end
436
437 "#{lhs} #{conditionTest}"
438 end
439
440 # conditionTest should contain a string that provides a comparator and a RHS
441 # value e.g. "< 0".
442 def cloopEmitTestAndBranchIf(operands, type, conditionTest, branchTarget)
443 conditionExpr = cloopGenerateConditionExpression(operands, type, conditionTest)
444 $asm.putc "if (#{conditionExpr})"
445 $asm.putc " goto #{branchTarget};"
446 end
447
448 def cloopEmitTestSet(operands, type, conditionTest)
449 # The result is a boolean condition. Hence, the result type is always an
450 # int. The passed in type is only used for the values being tested in
451 # the condition test.
452 conditionExpr = cloopGenerateConditionExpression(operands, type, conditionTest)
453 $asm.putc "#{operands[-1].clValue} = (#{conditionExpr});"
454 end
455
456 def cloopEmitOpAndBranch(operands, operator, type, conditionTest)
457 case type
458 when :int; tempType = "intptr_t"
459 when :int32; tempType = "int32_t"
460 when :int64; tempType = "int64_t"
461 else
462 raise "Unimplemented type"
463 end
464
465 op1 = operands[0].clValue(type)
466 op2 = operands[1].clValue(type)
467
468 $asm.putc "{"
469 $asm.putc " #{tempType} temp = #{op2} #{operator} #{op1};"
470 $asm.putc " #{op2} = temp;"
471 $asm.putc " if (temp #{conditionTest})"
472 $asm.putc " goto #{operands[2].cLabel};"
473 $asm.putc "}"
474 end
475
476 def cloopAddOverflowTest(operands, type)
477 case type
478 when :int32
479 tempType = "int32_t"
480 signBit = "SIGN_BIT32"
481 else
482 raise "Unimplemented type"
483 end
484
485 $asm.putc " #{tempType} a = #{operands[0].clValue(type)};"
486 $asm.putc " #{tempType} b = #{operands[1].clValue(type)};"
487 $asm.putc " // sign(b) sign(a) | Overflows if:"
488 $asm.putc " // 0 0 | sign(b+a) = 1 (pos + pos != neg)"
489 $asm.putc " // 0 1 | never"
490 $asm.putc " // 1 0 | never"
491 $asm.putc " // 1 1 | sign(b+a) = 0 (neg + neg != pos)"
492 "((#{signBit}(b) == #{signBit}(a)) && (#{signBit}(b+a) != #{signBit}(a)))"
493 end
494
495 def cloopSubOverflowTest(operands, type)
496 case type
497 when :int32
498 tempType = "int32_t"
499 signBit = "SIGN_BIT32"
500 else
501 raise "Unimplemented type"
502 end
503
504 $asm.putc " #{tempType} a = #{operands[0].clValue(type)};"
505 $asm.putc " #{tempType} b = #{operands[1].clValue(type)};"
506 $asm.putc " // sign(b) sign(a) | Overflows if:"
507 $asm.putc " // 0 0 | never"
508 $asm.putc " // 0 1 | sign(b-a) = 1 (pos - neg != pos)"
509 $asm.putc " // 1 0 | sign(b-a) = 0 (neg - pos != pos)"
510 $asm.putc " // 1 1 | never"
511 "((#{signBit}(b) != #{signBit}(a)) && (#{signBit}(b-a) == #{signBit}(a)))"
512 end
513
514 def cloopMulOverflowTest(operands, type)
515 case type
516 when :int32
517 tempType = "uint32_t"
518 else
519 raise "Unimplemented type"
520 end
521 $asm.putc " #{tempType} a = #{operands[0].clValue(type)};"
522 $asm.putc " #{tempType} b = #{operands[1].clValue(type)};"
523 "((b | a) >> 15)"
524 end
525
526 def cloopEmitOpAndBranchIfOverflow(operands, operator, type)
527 $asm.putc "{"
528
529 # Emit the overflow test based on the operands and the type:
530 case operator
531 when "+"; overflowTest = cloopAddOverflowTest(operands, type)
532 when "-"; overflowTest = cloopSubOverflowTest(operands, type)
533 when "*"; overflowTest = cloopMulOverflowTest(operands, type)
534 else
535 raise "Unimplemented opeartor"
536 end
537
538 $asm.putc " bool didOverflow = #{overflowTest};"
539 $asm.putc " #{operands[1].clValue(type)} = #{operands[1].clValue(type)} #{operator} #{operands[0].clValue(type)};"
540 $asm.putc " if (didOverflow)"
541 $asm.putc " goto #{operands[2].cLabel};"
542 $asm.putc "}"
543 end
544
545 # operands: callTarget, currentFrame, currentPC
546 def cloopEmitCallSlowPath(operands)
547 $asm.putc "{"
548 $asm.putc " SlowPathReturnType result = #{operands[0].cLabel}(#{operands[1].clDump}, #{operands[2].clDump});"
549 $asm.putc " decodeResult(result, t0.vp, t1.vp);"
550 $asm.putc "}"
551 end
552
553 def cloopEmitCallSlowPathVoid(operands)
554 $asm.putc "#{operands[0].cLabel}(#{operands[1].clDump}, #{operands[2].clDump});"
555 end
556
557 class Instruction
558 @@didReturnFromJSLabelCounter = 0
559
560 def lowerC_LOOP
561 $asm.codeOrigin codeOriginString if $enableCodeOriginComments
562 $asm.annotation annotation if $enableInstrAnnotations && (opcode != "cloopDo")
563
564 case opcode
565 when "addi"
566 cloopEmitOperation(operands, :int32, "+")
567 when "addq"
568 cloopEmitOperation(operands, :int64, "+")
569 when "addp"
570 cloopEmitOperation(operands, :int, "+")
571
572 when "andi"
573 cloopEmitOperation(operands, :int32, "&")
574 when "andq"
575 cloopEmitOperation(operands, :int64, "&")
576 when "andp"
577 cloopEmitOperation(operands, :int, "&")
578
579 when "ori"
580 cloopEmitOperation(operands, :int32, "|")
581 when "orq"
582 cloopEmitOperation(operands, :int64, "|")
583 when "orp"
584 cloopEmitOperation(operands, :int, "|")
585
586 when "xori"
587 cloopEmitOperation(operands, :int32, "^")
588 when "xorq"
589 cloopEmitOperation(operands, :int64, "^")
590 when "xorp"
591 cloopEmitOperation(operands, :int, "^")
592
593 when "lshifti"
594 cloopEmitShiftOperation(operands, :int32, "<<")
595 when "lshiftq"
596 cloopEmitShiftOperation(operands, :int64, "<<")
597 when "lshiftp"
598 cloopEmitShiftOperation(operands, :int, "<<")
599
600 when "rshifti"
601 cloopEmitShiftOperation(operands, :int32, ">>")
602 when "rshiftq"
603 cloopEmitShiftOperation(operands, :int64, ">>")
604 when "rshiftp"
605 cloopEmitShiftOperation(operands, :int, ">>")
606
607 when "urshifti"
608 cloopEmitShiftOperation(operands, :uint32, ">>")
609 when "urshiftq"
610 cloopEmitShiftOperation(operands, :uint64, ">>")
611 when "urshiftp"
612 cloopEmitShiftOperation(operands, :uint, ">>")
613
614 when "muli"
615 cloopEmitOperation(operands, :int32, "*")
616 when "mulq"
617 cloopEmitOperation(operands, :int64, "*")
618 when "mulp"
619 cloopEmitOperation(operands, :int, "*")
620
621 when "subi"
622 cloopEmitOperation(operands, :int32, "-")
623 when "subq"
624 cloopEmitOperation(operands, :int64, "-")
625 when "subp"
626 cloopEmitOperation(operands, :int, "-")
627
628 when "negi"
629 cloopEmitUnaryOperation(operands, :int32, "-")
630 when "negq"
631 cloopEmitUnaryOperation(operands, :int64, "-")
632 when "negp"
633 cloopEmitUnaryOperation(operands, :int, "-")
634
635 when "noti"
636 cloopEmitUnaryOperation(operands, :int32, "!")
637
638 when "loadi"
639 $asm.putc "#{operands[1].clValue(:uint)} = #{operands[0].uint32MemRef};"
640 # There's no need to call clearHighWord() here because the above will
641 # automatically take care of 0 extension.
642 when "loadis"
643 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].int32MemRef};"
644 when "loadq"
645 $asm.putc "#{operands[1].clValue(:int64)} = #{operands[0].int64MemRef};"
646 when "loadp"
647 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].intMemRef};"
648 when "storei"
649 $asm.putc "#{operands[1].int32MemRef} = #{operands[0].clValue(:int32)};"
650 when "storeq"
651 $asm.putc "#{operands[1].int64MemRef} = #{operands[0].clValue(:int64)};"
652 when "storep"
653 $asm.putc "#{operands[1].intMemRef} = #{operands[0].clValue(:int)};"
654 when "loadb"
655 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].uint8MemRef};"
656 when "loadbs"
657 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].int8MemRef};"
658 when "storeb"
659 $asm.putc "#{operands[1].uint8MemRef} = #{operands[0].clValue(:int8)};"
660 when "loadh"
661 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].uint16MemRef};"
662 when "loadhs"
663 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].int16MemRef};"
664 when "storeh"
665 $asm.putc "*#{operands[1].uint16MemRef} = #{operands[0].clValue(:int16)};"
666 when "loadd"
667 $asm.putc "#{operands[1].clValue(:double)} = #{operands[0].dblMemRef};"
668 when "stored"
669 $asm.putc "#{operands[1].dblMemRef} = #{operands[0].clValue(:double)};"
670
671 when "addd"
672 cloopEmitOperation(operands, :double, "+")
673 when "divd"
674 cloopEmitOperation(operands, :double, "/")
675 when "subd"
676 cloopEmitOperation(operands, :double, "-")
677 when "muld"
678 cloopEmitOperation(operands, :double, "*")
679
680 # Convert an int value to its double equivalent, and store it in a double register.
681 when "ci2d"
682 $asm.putc "#{operands[1].clValue(:double)} = #{operands[0].clValue(:int32)};"
683
684 when "bdeq"
685 cloopEmitCompareAndBranch(operands, :double, "==")
686 when "bdneq"
687 cloopEmitCompareAndBranch(operands, :double, "!=")
688 when "bdgt"
689 cloopEmitCompareAndBranch(operands, :double, ">");
690 when "bdgteq"
691 cloopEmitCompareAndBranch(operands, :double, ">=");
692 when "bdlt"
693 cloopEmitCompareAndBranch(operands, :double, "<");
694 when "bdlteq"
695 cloopEmitCompareAndBranch(operands, :double, "<=");
696
697 when "bdequn"
698 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "==")
699 when "bdnequn"
700 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "!=")
701 when "bdgtun"
702 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, ">")
703 when "bdgtequn"
704 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, ">=")
705 when "bdltun"
706 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "<")
707 when "bdltequn"
708 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "<=")
709
710 when "td2i"
711 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].clValue(:double)};"
712 $asm.putc "#{operands[1].clDump}.clearHighWord();"
713
714 when "bcd2i" # operands: srcDbl dstInt slowPath
715 $asm.putc "{"
716 $asm.putc " double d = #{operands[0].clValue(:double)};"
717 $asm.putc " const int32_t asInt32 = int32_t(d);"
718 $asm.putc " if (asInt32 != d || (!asInt32 && std::signbit(d))) // true for -0.0"
719 $asm.putc " goto #{operands[2].cLabel};"
720 $asm.putc " #{operands[1].clValue} = asInt32;"
721 $asm.putc " #{operands[1].clDump}.clearHighWord();"
722 $asm.putc "}"
723
724 when "move"
725 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].clValue(:int)};"
726 when "sxi2q"
727 $asm.putc "#{operands[1].clValue(:int64)} = #{operands[0].clValue(:int32)};"
728 when "zxi2q"
729 $asm.putc "#{operands[1].clValue(:uint64)} = #{operands[0].clValue(:uint32)};"
730 when "nop"
731 $asm.putc "// nop"
732 when "bbeq"
733 cloopEmitCompareAndBranch(operands, :int8, "==")
734 when "bieq"
735 cloopEmitCompareAndBranch(operands, :int32, "==")
736 when "bqeq"
737 cloopEmitCompareAndBranch(operands, :int64, "==")
738 when "bpeq"
739 cloopEmitCompareAndBranch(operands, :int, "==")
740
741 when "bbneq"
742 cloopEmitCompareAndBranch(operands, :int8, "!=")
743 when "bineq"
744 cloopEmitCompareAndBranch(operands, :int32, "!=")
745 when "bqneq"
746 cloopEmitCompareAndBranch(operands, :int64, "!=")
747 when "bpneq"
748 cloopEmitCompareAndBranch(operands, :int, "!=")
749
750 when "bba"
751 cloopEmitCompareAndBranch(operands, :uint8, ">")
752 when "bia"
753 cloopEmitCompareAndBranch(operands, :uint32, ">")
754 when "bqa"
755 cloopEmitCompareAndBranch(operands, :uint64, ">")
756 when "bpa"
757 cloopEmitCompareAndBranch(operands, :uint, ">")
758
759 when "bbaeq"
760 cloopEmitCompareAndBranch(operands, :uint8, ">=")
761 when "biaeq"
762 cloopEmitCompareAndBranch(operands, :uint32, ">=")
763 when "bqaeq"
764 cloopEmitCompareAndBranch(operands, :uint64, ">=")
765 when "bpaeq"
766 cloopEmitCompareAndBranch(operands, :uint, ">=")
767
768 when "bbb"
769 cloopEmitCompareAndBranch(operands, :uint8, "<")
770 when "bib"
771 cloopEmitCompareAndBranch(operands, :uint32, "<")
772 when "bqb"
773 cloopEmitCompareAndBranch(operands, :uint64, "<")
774 when "bpb"
775 cloopEmitCompareAndBranch(operands, :uint, "<")
776
777 when "bbbeq"
778 cloopEmitCompareAndBranch(operands, :uint8, "<=")
779 when "bibeq"
780 cloopEmitCompareAndBranch(operands, :uint32, "<=")
781 when "bqbeq"
782 cloopEmitCompareAndBranch(operands, :uint64, "<=")
783 when "bpbeq"
784 cloopEmitCompareAndBranch(operands, :uint, "<=")
785
786 when "bbgt"
787 cloopEmitCompareAndBranch(operands, :int8, ">")
788 when "bigt"
789 cloopEmitCompareAndBranch(operands, :int32, ">")
790 when "bqgt"
791 cloopEmitCompareAndBranch(operands, :int64, ">")
792 when "bpgt"
793 cloopEmitCompareAndBranch(operands, :int, ">")
794
795 when "bbgteq"
796 cloopEmitCompareAndBranch(operands, :int8, ">=")
797 when "bigteq"
798 cloopEmitCompareAndBranch(operands, :int32, ">=")
799 when "bqgteq"
800 cloopEmitCompareAndBranch(operands, :int64, ">=")
801 when "bpgteq"
802 cloopEmitCompareAndBranch(operands, :int, ">=")
803
804 when "bblt"
805 cloopEmitCompareAndBranch(operands, :int8, "<")
806 when "bilt"
807 cloopEmitCompareAndBranch(operands, :int32, "<")
808 when "bqlt"
809 cloopEmitCompareAndBranch(operands, :int64, "<")
810 when "bplt"
811 cloopEmitCompareAndBranch(operands, :int, "<")
812
813 when "bblteq"
814 cloopEmitCompareAndBranch(operands, :int8, "<=")
815 when "bilteq"
816 cloopEmitCompareAndBranch(operands, :int32, "<=")
817 when "bqlteq"
818 cloopEmitCompareAndBranch(operands, :int64, "<=")
819 when "bplteq"
820 cloopEmitCompareAndBranch(operands, :int, "<=")
821
822 when "btbz"
823 cloopEmitTestAndBranchIf(operands, :int8, "== 0", operands[-1].cLabel)
824 when "btiz"
825 cloopEmitTestAndBranchIf(operands, :int32, "== 0", operands[-1].cLabel)
826 when "btqz"
827 cloopEmitTestAndBranchIf(operands, :int64, "== 0", operands[-1].cLabel)
828 when "btpz"
829 cloopEmitTestAndBranchIf(operands, :int, "== 0", operands[-1].cLabel)
830
831 when "btbnz"
832 cloopEmitTestAndBranchIf(operands, :int8, "!= 0", operands[-1].cLabel)
833 when "btinz"
834 cloopEmitTestAndBranchIf(operands, :int32, "!= 0", operands[-1].cLabel)
835 when "btqnz"
836 cloopEmitTestAndBranchIf(operands, :int64, "!= 0", operands[-1].cLabel)
837 when "btpnz"
838 cloopEmitTestAndBranchIf(operands, :int, "!= 0", operands[-1].cLabel)
839
840 when "btbs"
841 cloopEmitTestAndBranchIf(operands, :int8, "< 0", operands[-1].cLabel)
842 when "btis"
843 cloopEmitTestAndBranchIf(operands, :int32, "< 0", operands[-1].cLabel)
844 when "btqs"
845 cloopEmitTestAndBranchIf(operands, :int64, "< 0", operands[-1].cLabel)
846 when "btps"
847 cloopEmitTestAndBranchIf(operands, :int, "< 0", operands[-1].cLabel)
848
849 # For jmp, we do not want to assume that we have COMPUTED_GOTO support.
850 # Fortunately, the only times we should ever encounter indirect jmps is
851 # when the jmp target is a CLoop opcode (by design).
852 #
853 # Hence, we check if the jmp target is a known label reference. If so,
854 # we can emit a goto directly. If it is not a known target, then we set
855 # the target in the opcode, and dispatch to it via whatever dispatch
856 # mechanism is in used.
857 when "jmp"
858 if operands[0].is_a? LocalLabelReference or operands[0].is_a? LabelReference
859 # Handles jumps local or global labels.
860 $asm.putc "goto #{operands[0].cLabel};"
861 else
862 # Handles jumps to some computed target.
863 # NOTE: must be an opcode handler or a llint glue helper.
864 $asm.putc "opcode = #{operands[0].clValue(:opcode)};"
865 $asm.putc "DISPATCH_OPCODE();"
866 end
867
868 when "call"
869 $asm.putc "CRASH(); // generic call instruction not supported by design!"
870 when "break"
871 $asm.putc "CRASH(); // break instruction not implemented."
872 when "ret"
873 $asm.putc "opcode = lr.opcode;"
874 $asm.putc "DISPATCH_OPCODE();"
875
876 when "cbeq"
877 cloopEmitCompareAndSet(operands, :uint8, "==")
878 when "cieq"
879 cloopEmitCompareAndSet(operands, :uint32, "==")
880 when "cqeq"
881 cloopEmitCompareAndSet(operands, :uint64, "==")
882 when "cpeq"
883 cloopEmitCompareAndSet(operands, :uint, "==")
884
885 when "cbneq"
886 cloopEmitCompareAndSet(operands, :uint8, "!=")
887 when "cineq"
888 cloopEmitCompareAndSet(operands, :uint32, "!=")
889 when "cqneq"
890 cloopEmitCompareAndSet(operands, :uint64, "!=")
891 when "cpneq"
892 cloopEmitCompareAndSet(operands, :uint, "!=")
893
894 when "cba"
895 cloopEmitCompareAndSet(operands, :uint8, ">")
896 when "cia"
897 cloopEmitCompareAndSet(operands, :uint32, ">")
898 when "cqa"
899 cloopEmitCompareAndSet(operands, :uint64, ">")
900 when "cpa"
901 cloopEmitCompareAndSet(operands, :uint, ">")
902
903 when "cbaeq"
904 cloopEmitCompareAndSet(operands, :uint8, ">=")
905 when "ciaeq"
906 cloopEmitCompareAndSet(operands, :uint32, ">=")
907 when "cqaeq"
908 cloopEmitCompareAndSet(operands, :uint64, ">=")
909 when "cpaeq"
910 cloopEmitCompareAndSet(operands, :uint, ">=")
911
912 when "cbb"
913 cloopEmitCompareAndSet(operands, :uint8, "<")
914 when "cib"
915 cloopEmitCompareAndSet(operands, :uint32, "<")
916 when "cqb"
917 cloopEmitCompareAndSet(operands, :uint64, "<")
918 when "cpb"
919 cloopEmitCompareAndSet(operands, :uint, "<")
920
921 when "cbbeq"
922 cloopEmitCompareAndSet(operands, :uint8, "<=")
923 when "cibeq"
924 cloopEmitCompareAndSet(operands, :uint32, "<=")
925 when "cqbeq"
926 cloopEmitCompareAndSet(operands, :uint64, "<=")
927 when "cpbeq"
928 cloopEmitCompareAndSet(operands, :uint, "<=")
929
930 when "cbgt"
931 cloopEmitCompareAndSet(operands, :int8, ">")
932 when "cigt"
933 cloopEmitCompareAndSet(operands, :int32, ">")
934 when "cqgt"
935 cloopEmitCompareAndSet(operands, :int64, ">")
936 when "cpgt"
937 cloopEmitCompareAndSet(operands, :int, ">")
938
939 when "cbgteq"
940 cloopEmitCompareAndSet(operands, :int8, ">=")
941 when "cigteq"
942 cloopEmitCompareAndSet(operands, :int32, ">=")
943 when "cqgteq"
944 cloopEmitCompareAndSet(operands, :int64, ">=")
945 when "cpgteq"
946 cloopEmitCompareAndSet(operands, :int, ">=")
947
948 when "cblt"
949 cloopEmitCompareAndSet(operands, :int8, "<")
950 when "cilt"
951 cloopEmitCompareAndSet(operands, :int32, "<")
952 when "cqlt"
953 cloopEmitCompareAndSet(operands, :int64, "<")
954 when "cplt"
955 cloopEmitCompareAndSet(operands, :int, "<")
956
957 when "cblteq"
958 cloopEmitCompareAndSet(operands, :int8, "<=")
959 when "cilteq"
960 cloopEmitCompareAndSet(operands, :int32, "<=")
961 when "cqlteq"
962 cloopEmitCompareAndSet(operands, :int64, "<=")
963 when "cplteq"
964 cloopEmitCompareAndSet(operands, :int, "<=")
965
966 when "tbs"
967 cloopEmitTestSet(operands, :int8, "< 0")
968 when "tis"
969 cloopEmitTestSet(operands, :int32, "< 0")
970 when "tqs"
971 cloopEmitTestSet(operands, :int64, "< 0")
972 when "tps"
973 cloopEmitTestSet(operands, :int, "< 0")
974
975 when "tbz"
976 cloopEmitTestSet(operands, :int8, "== 0")
977 when "tiz"
978 cloopEmitTestSet(operands, :int32, "== 0")
979 when "tqz"
980 cloopEmitTestSet(operands, :int64, "== 0")
981 when "tpz"
982 cloopEmitTestSet(operands, :int, "== 0")
983
984 when "tbnz"
985 cloopEmitTestSet(operands, :int8, "!= 0")
986 when "tinz"
987 cloopEmitTestSet(operands, :int32, "!= 0")
988 when "tqnz"
989 cloopEmitTestSet(operands, :int64, "!= 0")
990 when "tpnz"
991 cloopEmitTestSet(operands, :int, "!= 0")
992
993 # 64-bit instruction: cdqi (based on X64)
994 # Sign extends the lower 32 bits of t0, but put the sign extension into
995 # the lower 32 bits of t1. Leave the upper 32 bits of t0 and t1 unchanged.
996 when "cdqi"
997 $asm.putc "{"
998 $asm.putc " int64_t temp = t0.i32; // sign extend the low 32bit"
999 $asm.putc " t0.i32 = temp; // low word"
1000 $asm.putc " t0.clearHighWord();"
1001 $asm.putc " t1.i32 = uint64_t(temp) >> 32; // high word"
1002 $asm.putc " t1.clearHighWord();"
1003 $asm.putc "}"
1004
1005 # 64-bit instruction: idivi op1 (based on X64)
1006 # Divide a 64-bit integer numerator by the specified denominator.
1007 # The numerator is specified in t0 and t1 as follows:
1008 # 1. low 32 bits of the numerator is in the low 32 bits of t0.
1009 # 2. high 32 bits of the numerator is in the low 32 bits of t1.
1010 #
1011 # The resultant quotient is a signed 32-bit int, and is to be stored
1012 # in the lower 32 bits of t0.
1013 # The resultant remainder is a signed 32-bit int, and is to be stored
1014 # in the lower 32 bits of t1.
1015 when "idivi"
1016 # Divide t1,t0 (EDX,EAX) by the specified arg, and store the remainder in t1,
1017 # and quotient in t0:
1018 $asm.putc "{"
1019 $asm.putc " int64_t dividend = (int64_t(t1.u32) << 32) | t0.u32;"
1020 $asm.putc " int64_t divisor = #{operands[0].clValue(:int)};"
1021 $asm.putc " t1.i32 = dividend % divisor; // remainder"
1022 $asm.putc " t1.clearHighWord();"
1023 $asm.putc " t0.i32 = dividend / divisor; // quotient"
1024 $asm.putc " t0.clearHighWord();"
1025 $asm.putc "}"
1026
1027 # 32-bit instruction: fii2d int32LoOp int32HiOp dblOp (based on ARMv7)
1028 # Decode 2 32-bit ints (low and high) into a 64-bit double.
1029 when "fii2d"
1030 $asm.putc "#{operands[2].clValue(:double)} = Ints2Double(#{operands[0].clValue(:uint32)}, #{operands[1].clValue(:uint32)});"
1031
1032 # 32-bit instruction: f2dii dblOp int32LoOp int32HiOp (based on ARMv7)
1033 # Encode a 64-bit double into 2 32-bit ints (low and high).
1034 when "fd2ii"
1035 $asm.putc "Double2Ints(#{operands[0].clValue(:double)}, #{operands[1].clValue(:uint32)}, #{operands[2].clValue(:uint32)});"
1036
1037 # 64-bit instruction: fq2d int64Op dblOp (based on X64)
1038 # Copy a bit-encoded double in a 64-bit int register to a double register.
1039 when "fq2d"
1040 $asm.putc "#{operands[1].clValue(:double)} = #{operands[0].clValue(:castToDouble)};"
1041
1042 # 64-bit instruction: fd2q dblOp int64Op (based on X64 instruction set)
1043 # Copy a double as a bit-encoded double into a 64-bit int register.
1044 when "fd2q"
1045 $asm.putc "#{operands[1].clValue(:int64)} = #{operands[0].clValue(:castToInt64)};"
1046
1047 when "leai"
1048 operands[0].cloopEmitLea(operands[1], :int32)
1049 when "leap"
1050 operands[0].cloopEmitLea(operands[1], :int)
1051
1052 when "baddio"
1053 cloopEmitOpAndBranchIfOverflow(operands, "+", :int32)
1054 when "bsubio"
1055 cloopEmitOpAndBranchIfOverflow(operands, "-", :int32)
1056 when "bmulio"
1057 cloopEmitOpAndBranchIfOverflow(operands, "*", :int32)
1058
1059 when "baddis"
1060 cloopEmitOpAndBranch(operands, "+", :int32, "< 0")
1061 when "baddiz"
1062 cloopEmitOpAndBranch(operands, "+", :int32, "== 0")
1063 when "baddinz"
1064 cloopEmitOpAndBranch(operands, "+", :int32, "!= 0")
1065
1066 when "baddqs"
1067 cloopEmitOpAndBranch(operands, "+", :int64, "< 0")
1068 when "baddqz"
1069 cloopEmitOpAndBranch(operands, "+", :int64, "== 0")
1070 when "baddqnz"
1071 cloopEmitOpAndBranch(operands, "+", :int64, "!= 0")
1072
1073 when "baddps"
1074 cloopEmitOpAndBranch(operands, "+", :int, "< 0")
1075 when "baddpz"
1076 cloopEmitOpAndBranch(operands, "+", :int, "== 0")
1077 when "baddpnz"
1078 cloopEmitOpAndBranch(operands, "+", :int, "!= 0")
1079
1080 when "bsubis"
1081 cloopEmitOpAndBranch(operands, "-", :int32, "< 0")
1082 when "bsubiz"
1083 cloopEmitOpAndBranch(operands, "-", :int32, "== 0")
1084 when "bsubinz"
1085 cloopEmitOpAndBranch(operands, "-", :int32, "!= 0")
1086
1087 when "borris"
1088 cloopEmitOpAndBranch(operands, "|", :int32, "< 0")
1089 when "borriz"
1090 cloopEmitOpAndBranch(operands, "|", :int32, "== 0")
1091 when "borrinz"
1092 cloopEmitOpAndBranch(operands, "|", :int32, "!= 0")
1093
1094 when "memfence"
1095
1096 when "push"
1097 operands.each {
1098 | op |
1099 $asm.putc "PUSH(#{op.clDump});"
1100 }
1101 when "pop"
1102 operands.each {
1103 | op |
1104 $asm.putc "POP(#{op.clDump});"
1105 }
1106
1107 when "pushCalleeSaves"
1108 when "popCalleeSaves"
1109
1110
1111 # A convenience and compact call to crash because we don't want to use
1112 # the generic llint crash mechanism which relies on the availability
1113 # of the call instruction (which cannot be implemented in a generic
1114 # way, and can be abused if we made it just work for this special case).
1115 # Using a special cloopCrash instruction is cleaner.
1116 when "cloopCrash"
1117 $asm.putc "CRASH();"
1118
1119 # We can't rely on the llint JS call mechanism which actually makes
1120 # use of the call instruction. Instead, we just implement JS calls
1121 # as an opcode dispatch.
1122 when "cloopCallJSFunction"
1123 @@didReturnFromJSLabelCounter += 1
1124 $asm.putc "lr.opcode = getOpcode(llint_cloop_did_return_from_js_#{@@didReturnFromJSLabelCounter});"
1125 $asm.putc "opcode = #{operands[0].clValue(:opcode)};"
1126 $asm.putc "DISPATCH_OPCODE();"
1127 $asm.putsLabel("llint_cloop_did_return_from_js_#{@@didReturnFromJSLabelCounter}", false)
1128
1129 # We can't do generic function calls with an arbitrary set of args, but
1130 # fortunately we don't have to here. All native function calls always
1131 # have a fixed prototype of 1 args: the passed ExecState.
1132 when "cloopCallNative"
1133 $asm.putc "nativeFunc = #{operands[0].clValue(:nativeFunc)};"
1134 $asm.putc "functionReturnValue = JSValue::decode(nativeFunc(t0.execState));"
1135 $asm.putc "#if USE(JSVALUE32_64)"
1136 $asm.putc " t1.i = functionReturnValue.tag();"
1137 $asm.putc " t0.i = functionReturnValue.payload();"
1138 $asm.putc "#else // USE_JSVALUE64)"
1139 $asm.putc " t0.encodedJSValue = JSValue::encode(functionReturnValue);"
1140 $asm.putc "#endif // USE_JSVALUE64)"
1141
1142 # We can't do generic function calls with an arbitrary set of args, but
1143 # fortunately we don't have to here. All slow path function calls always
1144 # have a fixed prototype too. See cloopEmitCallSlowPath() for details.
1145 when "cloopCallSlowPath"
1146 cloopEmitCallSlowPath(operands)
1147
1148 when "cloopCallSlowPathVoid"
1149 cloopEmitCallSlowPathVoid(operands)
1150
1151 # For debugging only. This is used to insert instrumentation into the
1152 # generated LLIntAssembly.h during llint development only. Do not use
1153 # for production code.
1154 when "cloopDo"
1155 $asm.putc "#{annotation}"
1156
1157 else
1158 lowerDefault
1159 end
1160 end
1161 end