1 # Copyright (C) 2012, 2014 Apple Inc. All rights reserved.
3 # Redistribution and use in source and binary forms, with or without
4 # modification, are permitted provided that the following conditions
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.
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.
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
34 def cloopMapType(type
)
46 when :nativeFunc; ".nativeFunc"
48 when :castToDouble; ".castToDouble"
49 when :castToInt64; ".castToInt64"
50 when :opcode; ".opcode"
52 raise "Unsupported type"
57 class SpecialRegister
< NoChildren
61 def clValue(type
=:int)
62 @name +
cloopMapType(type
)
66 C_LOOP_SCRATCH_FPR
= SpecialRegister
.new("d6")
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.
100 raise "Bad register #{name} for C_LOOP at #{codeOriginString}"
103 def clValue(type
=:int)
104 clDump +
cloopMapType(type
)
124 raise "Bad register #{name} for C_LOOP at #{codeOriginString}"
127 def clValue(type
=:int)
128 clDump +
cloopMapType(type
)
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.
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.
147 valueStr
= (value
< 0) ? "#{value}" : "0x#{value.to_s(16)}"
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})"
159 raise "Not implemented immediate of type: #{type}"
166 "[#{base.clDump}, #{offset.value}]"
168 def clValue(type
=:int)
170 when :int8; int8MemRef
171 when :int32; int32MemRef
172 when :int64; int64MemRef
174 when :uint8; uint8MemRef
175 when :uint32; uint32MemRef
176 when :uint64; uint64MemRef
177 when :uint; uintMemRef
178 when :opcode; opcodeMemRef
179 when :nativeFunc; nativeFuncMemRef
181 raise "Unexpected Address type: #{type}"
186 "#{base.clValue(:int8Ptr)}"
187 elsif offset
.value
> 0
188 "#{base.clValue(:int8Ptr)} + #{offset.value}"
190 "#{base.clValue(:int8Ptr)} - #{-offset.value}"
194 "*CAST
<int8_t
*>(#{pointerExpr})"
197 "*CAST
<int16_t
*>(#{pointerExpr})"
200 "*CAST
<int32_t
*>(#{pointerExpr})"
203 "*CAST
<int64_t
*>(#{pointerExpr})"
206 "*CAST
<intptr_t
*>(#{pointerExpr})"
209 "*CAST
<uint8_t
*>(#{pointerExpr})"
212 "*CAST
<uint16_t
*>(#{pointerExpr})"
215 "*CAST
<uint32_t
*>(#{pointerExpr})"
218 "*CAST
<uint64_t
*>(#{pointerExpr})"
221 "*CAST
<uintptr_t
*>(#{pointerExpr})"
224 "*CAST
<NativeFunction
*>(#{pointerExpr})"
227 "*CAST
<Opcode
*>(#{pointerExpr})"
230 "*CAST
<double
*>(#{pointerExpr})"
236 "[#{base.clDump}, #{offset.clDump}, #{index.clDump} << #{scaleShift}]"
238 def clValue(type=:int)
240 when :int8; int8MemRef
241 when :int32; int32MemRef
242 when :int64; int64MemRef
244 when :uint8; uint8MemRef
245 when :uint32; uint32MemRef
246 when :uint64; uint64MemRef
247 when :uint; uintMemRef
248 when :opcode; opcodeMemRef
250 raise "Unexpected BaseIndex type
: #{type}"
255 "#{base.clValue(:int8Ptr)} +
(#{index.clValue} << #{scaleShift})"
257 "#{base.clValue(:int8Ptr)} +
(#{index.clValue} << #{scaleShift}) +
#{offset.clValue}"
261 "*CAST
<int8_t
*>(#{pointerExpr})"
264 "*CAST
<int16_t
*>(#{pointerExpr})"
267 "*CAST
<int32_t
*>(#{pointerExpr})"
270 "*CAST
<int64_t
*>(#{pointerExpr})"
273 "*CAST
<intptr_t
*>(#{pointerExpr})"
276 "*CAST
<uint8_t
*>(#{pointerExpr})"
279 "*CAST
<uint16_t
*>(#{pointerExpr})"
282 "*CAST
<uint32_t
*>(#{pointerExpr})"
285 "*CAST
<uint64_t
*>(#{pointerExpr})"
288 "*CAST
<uintptr_t
*>(#{pointerExpr})"
291 "*CAST
<Opcode
*>(#{pointerExpr})"
294 "*CAST
<double
*>(#{pointerExpr})"
298 class AbsoluteAddress
300 "#{codeOriginString}"
313 def cloopEmitLea(destination, type)
314 if destination == base
315 $asm.putc "#{destination.clValue(:int8Ptr)} +
= #{offset.clValue(type)};"
317 $asm.putc "#{destination.clValue(:int8Ptr)} = #{base.clValue(:int8Ptr)} +
#{offset.clValue(type)};"
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});"
330 # Actual lowering code follows.
334 def getModifiedListC_LOOP
337 # Verify that we will only see instructions and labels.
340 unless node.is_a? Instruction or
342 node.is_a? LocalLabel or
344 raise "Unexpected
#{node.inspect} at
#{node.codeOrigin}"
352 def clOperands(operands)
353 operands.map{|v| v.clDump}.join(", ")
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.
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.
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.
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.
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.
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};"
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)});"
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};"
422 # conditionTest should contain a string that provides a comparator and a RHS
424 def cloopGenerateConditionExpression(operands, type, conditionTest)
425 op1 = operands[0].clValue(type)
427 # The operands must consist of 2 or 3 values.
429 when 2 # Just test op1 against the conditionTest.
431 when 3 # Mask op1 with op2 before testing against the conditionTest.
432 lhs = "(#{op1} & #{operands[1].clValue(type)})"
434 raise "Expected
2 or 3 operands but got
#{operands.size} at
#{codeOriginString}"
437 "#{lhs} #{conditionTest}"
440 # conditionTest should contain a string that provides a comparator and a RHS
442 def cloopEmitTestAndBranchIf(operands, type, conditionTest, branchTarget)
443 conditionExpr = cloopGenerateConditionExpression(operands, type, conditionTest)
444 $asm.putc "if (#{conditionExpr})"
445 $asm.putc " goto
#{branchTarget};"
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});"
456 def cloopEmitOpAndBranch(operands, operator, type, conditionTest)
458 when :int; tempType = "intptr_t
"
459 when :int32; tempType = "int32_t
"
460 when :int64; tempType = "int64_t
"
462 raise "Unimplemented type
"
465 op1 = operands[0].clValue(type)
466 op2 = operands[1].clValue(type)
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};"
476 def cloopAddOverflowTest(operands, type)
480 signBit = "SIGN_BIT32
"
482 raise "Unimplemented type
"
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
)))"
495 def cloopSubOverflowTest(operands, type)
499 signBit = "SIGN_BIT32
"
501 raise "Unimplemented type
"
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
)))"
514 def cloopMulOverflowTest(operands, type)
517 tempType = "uint32_t
"
519 raise "Unimplemented type
"
521 $asm.putc " #{tempType} a
= #{operands[0].clValue(type)};"
522 $asm.putc " #{tempType} b
= #{operands[1].clValue(type)};"
526 def cloopEmitOpAndBranchIfOverflow(operands, operator, type)
529 # Emit the overflow test based on the operands and the type:
531 when "+
"; overflowTest = cloopAddOverflowTest(operands, type)
532 when "-"; overflowTest = cloopSubOverflowTest(operands, type)
533 when "*"; overflowTest = cloopMulOverflowTest(operands, type)
535 raise "Unimplemented opeartor
"
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};"
545 # operands: callTarget, currentFrame, currentPC
546 def cloopEmitCallSlowPath(operands)
548 $asm.putc " SlowPathReturnType result
= #{operands[0].cLabel}(#{operands[1].clDump}, #{operands[2].clDump});"
549 $asm.putc " decodeResult(result
, t0
.vp
, t1
.vp
);"
553 def cloopEmitCallSlowPathVoid(operands)
554 $asm.putc "#{operands[0].cLabel}(#{operands[1].clDump}, #{operands[2].clDump});"
558 @@didReturnFromJSLabelCounter = 0
561 $asm.codeOrigin codeOriginString if $enableCodeOriginComments
562 $asm.annotation annotation if $enableInstrAnnotations && (opcode != "cloopDo")
566 cloopEmitOperation(operands
, :int32, "+")
568 cloopEmitOperation(operands, :int64, "+
")
570 cloopEmitOperation(operands, :int, "+
")
573 cloopEmitOperation(operands, :int32, "&")
575 cloopEmitOperation(operands, :int64, "&")
577 cloopEmitOperation(operands, :int, "&")
580 cloopEmitOperation(operands, :int32, "|")
582 cloopEmitOperation(operands, :int64, "|")
584 cloopEmitOperation(operands, :int, "|")
587 cloopEmitOperation(operands, :int32, "^
")
589 cloopEmitOperation(operands, :int64, "^
")
591 cloopEmitOperation(operands, :int, "^
")
594 cloopEmitShiftOperation(operands, :int32, "<<")
596 cloopEmitShiftOperation(operands, :int64, "<<")
598 cloopEmitShiftOperation(operands, :int, "<<")
601 cloopEmitShiftOperation(operands, :int32, ">>")
603 cloopEmitShiftOperation(operands, :int64, ">>")
605 cloopEmitShiftOperation(operands, :int, ">>")
608 cloopEmitShiftOperation(operands, :uint32, ">>")
610 cloopEmitShiftOperation(operands, :uint64, ">>")
612 cloopEmitShiftOperation(operands, :uint, ">>")
615 cloopEmitOperation(operands, :int32, "*")
617 cloopEmitOperation(operands, :int64, "*")
619 cloopEmitOperation(operands, :int, "*")
622 cloopEmitOperation(operands, :int32, "-")
624 cloopEmitOperation(operands, :int64, "-")
626 cloopEmitOperation(operands, :int, "-")
629 cloopEmitUnaryOperation(operands, :int32, "-")
631 cloopEmitUnaryOperation(operands, :int64, "-")
633 cloopEmitUnaryOperation(operands, :int, "-")
636 cloopEmitUnaryOperation(operands, :int32, "!
")
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.
643 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].int32MemRef};"
645 $asm.putc "#{operands[1].clValue(:int64)} = #{operands[0].int64MemRef};"
647 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].intMemRef};"
649 $asm.putc "#{operands[1].int32MemRef} = #{operands[0].clValue(:int32)};"
651 $asm.putc "#{operands[1].int64MemRef} = #{operands[0].clValue(:int64)};"
653 $asm.putc "#{operands[1].intMemRef} = #{operands[0].clValue(:int)};"
655 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].uint8MemRef};"
657 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].int8MemRef};"
659 $asm.putc "#{operands[1].uint8MemRef} = #{operands[0].clValue(:int8)};"
661 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].uint16MemRef};"
663 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].int16MemRef};"
665 $asm.putc "*#{operands[1].uint16MemRef} = #{operands[0].clValue(:int16)};"
667 $asm.putc "#{operands[1].clValue(:double)} = #{operands[0].dblMemRef};"
669 $asm.putc "#{operands[1].dblMemRef} = #{operands[0].clValue(:double)};"
672 cloopEmitOperation(operands, :double, "+
")
674 cloopEmitOperation(operands, :double, "/")
676 cloopEmitOperation(operands, :double, "-")
678 cloopEmitOperation(operands, :double, "*")
680 # Convert an int value to its double equivalent, and store it in a double register.
682 $asm.putc "#{operands[1].clValue(:double)} = #{operands[0].clValue(:int32)};"
685 cloopEmitCompareAndBranch(operands, :double, "==")
687 cloopEmitCompareAndBranch(operands, :double, "!
=")
689 cloopEmitCompareAndBranch(operands, :double, ">");
691 cloopEmitCompareAndBranch(operands, :double, ">=");
693 cloopEmitCompareAndBranch(operands, :double, "<");
695 cloopEmitCompareAndBranch(operands, :double, "<=");
698 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "==")
700 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "!
=")
702 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, ">")
704 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, ">=")
706 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "<")
708 cloopEmitCompareDoubleWithNaNCheckAndBranch(operands, "<=")
711 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].clValue(:double)};"
712 $asm.putc "#{operands[1].clDump}.clearHighWord();"
714 when "bcd2i
" # operands: srcDbl dstInt slowPath
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();"
725 $asm.putc "#{operands[1].clValue(:int)} = #{operands[0].clValue(:int)};"
727 $asm.putc "#{operands[1].clValue(:int64)} = #{operands[0].clValue(:int32)};"
729 $asm.putc "#{operands[1].clValue(:uint64)} = #{operands[0].clValue(:uint32)};"
733 cloopEmitCompareAndBranch(operands, :int8, "==")
735 cloopEmitCompareAndBranch(operands, :int32, "==")
737 cloopEmitCompareAndBranch(operands, :int64, "==")
739 cloopEmitCompareAndBranch(operands, :int, "==")
742 cloopEmitCompareAndBranch(operands, :int8, "!
=")
744 cloopEmitCompareAndBranch(operands, :int32, "!
=")
746 cloopEmitCompareAndBranch(operands, :int64, "!
=")
748 cloopEmitCompareAndBranch(operands, :int, "!
=")
751 cloopEmitCompareAndBranch(operands, :uint8, ">")
753 cloopEmitCompareAndBranch(operands, :uint32, ">")
755 cloopEmitCompareAndBranch(operands, :uint64, ">")
757 cloopEmitCompareAndBranch(operands, :uint, ">")
760 cloopEmitCompareAndBranch(operands, :uint8, ">=")
762 cloopEmitCompareAndBranch(operands, :uint32, ">=")
764 cloopEmitCompareAndBranch(operands, :uint64, ">=")
766 cloopEmitCompareAndBranch(operands, :uint, ">=")
769 cloopEmitCompareAndBranch(operands, :uint8, "<")
771 cloopEmitCompareAndBranch(operands, :uint32, "<")
773 cloopEmitCompareAndBranch(operands, :uint64, "<")
775 cloopEmitCompareAndBranch(operands, :uint, "<")
778 cloopEmitCompareAndBranch(operands, :uint8, "<=")
780 cloopEmitCompareAndBranch(operands, :uint32, "<=")
782 cloopEmitCompareAndBranch(operands, :uint64, "<=")
784 cloopEmitCompareAndBranch(operands, :uint, "<=")
787 cloopEmitCompareAndBranch(operands, :int8, ">")
789 cloopEmitCompareAndBranch(operands, :int32, ">")
791 cloopEmitCompareAndBranch(operands, :int64, ">")
793 cloopEmitCompareAndBranch(operands, :int, ">")
796 cloopEmitCompareAndBranch(operands, :int8, ">=")
798 cloopEmitCompareAndBranch(operands, :int32, ">=")
800 cloopEmitCompareAndBranch(operands, :int64, ">=")
802 cloopEmitCompareAndBranch(operands, :int, ">=")
805 cloopEmitCompareAndBranch(operands, :int8, "<")
807 cloopEmitCompareAndBranch(operands, :int32, "<")
809 cloopEmitCompareAndBranch(operands, :int64, "<")
811 cloopEmitCompareAndBranch(operands, :int, "<")
814 cloopEmitCompareAndBranch(operands, :int8, "<=")
816 cloopEmitCompareAndBranch(operands, :int32, "<=")
818 cloopEmitCompareAndBranch(operands, :int64, "<=")
820 cloopEmitCompareAndBranch(operands, :int, "<=")
823 cloopEmitTestAndBranchIf(operands, :int8, "== 0", operands[-1].cLabel)
825 cloopEmitTestAndBranchIf(operands, :int32, "== 0", operands[-1].cLabel)
827 cloopEmitTestAndBranchIf(operands, :int64, "== 0", operands[-1].cLabel)
829 cloopEmitTestAndBranchIf(operands, :int, "== 0", operands[-1].cLabel)
832 cloopEmitTestAndBranchIf(operands, :int8, "!
= 0", operands[-1].cLabel)
834 cloopEmitTestAndBranchIf(operands, :int32, "!
= 0", operands[-1].cLabel)
836 cloopEmitTestAndBranchIf(operands, :int64, "!
= 0", operands[-1].cLabel)
838 cloopEmitTestAndBranchIf(operands, :int, "!
= 0", operands[-1].cLabel)
841 cloopEmitTestAndBranchIf(operands, :int8, "< 0", operands[-1].cLabel)
843 cloopEmitTestAndBranchIf(operands, :int32, "< 0", operands[-1].cLabel)
845 cloopEmitTestAndBranchIf(operands, :int64, "< 0", operands[-1].cLabel)
847 cloopEmitTestAndBranchIf(operands, :int, "< 0", operands[-1].cLabel)
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).
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.
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};"
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();"
869 $asm.putc "CRASH(); // generic call instruction
not supported by design!
"
871 $asm.putc "CRASH(); // break instruction
not implemented
."
873 $asm.putc "opcode
= lr
.opcode
;"
874 $asm.putc "DISPATCH_OPCODE();"
877 cloopEmitCompareAndSet(operands, :uint8, "==")
879 cloopEmitCompareAndSet(operands, :uint32, "==")
881 cloopEmitCompareAndSet(operands, :uint64, "==")
883 cloopEmitCompareAndSet(operands, :uint, "==")
886 cloopEmitCompareAndSet(operands, :uint8, "!
=")
888 cloopEmitCompareAndSet(operands, :uint32, "!
=")
890 cloopEmitCompareAndSet(operands, :uint64, "!
=")
892 cloopEmitCompareAndSet(operands, :uint, "!
=")
895 cloopEmitCompareAndSet(operands, :uint8, ">")
897 cloopEmitCompareAndSet(operands, :uint32, ">")
899 cloopEmitCompareAndSet(operands, :uint64, ">")
901 cloopEmitCompareAndSet(operands, :uint, ">")
904 cloopEmitCompareAndSet(operands, :uint8, ">=")
906 cloopEmitCompareAndSet(operands, :uint32, ">=")
908 cloopEmitCompareAndSet(operands, :uint64, ">=")
910 cloopEmitCompareAndSet(operands, :uint, ">=")
913 cloopEmitCompareAndSet(operands, :uint8, "<")
915 cloopEmitCompareAndSet(operands, :uint32, "<")
917 cloopEmitCompareAndSet(operands, :uint64, "<")
919 cloopEmitCompareAndSet(operands, :uint, "<")
922 cloopEmitCompareAndSet(operands, :uint8, "<=")
924 cloopEmitCompareAndSet(operands, :uint32, "<=")
926 cloopEmitCompareAndSet(operands, :uint64, "<=")
928 cloopEmitCompareAndSet(operands, :uint, "<=")
931 cloopEmitCompareAndSet(operands, :int8, ">")
933 cloopEmitCompareAndSet(operands, :int32, ">")
935 cloopEmitCompareAndSet(operands, :int64, ">")
937 cloopEmitCompareAndSet(operands, :int, ">")
940 cloopEmitCompareAndSet(operands, :int8, ">=")
942 cloopEmitCompareAndSet(operands, :int32, ">=")
944 cloopEmitCompareAndSet(operands, :int64, ">=")
946 cloopEmitCompareAndSet(operands, :int, ">=")
949 cloopEmitCompareAndSet(operands, :int8, "<")
951 cloopEmitCompareAndSet(operands, :int32, "<")
953 cloopEmitCompareAndSet(operands, :int64, "<")
955 cloopEmitCompareAndSet(operands, :int, "<")
958 cloopEmitCompareAndSet(operands, :int8, "<=")
960 cloopEmitCompareAndSet(operands, :int32, "<=")
962 cloopEmitCompareAndSet(operands, :int64, "<=")
964 cloopEmitCompareAndSet(operands, :int, "<=")
967 cloopEmitTestSet(operands, :int8, "< 0")
969 cloopEmitTestSet(operands, :int32, "< 0")
971 cloopEmitTestSet(operands, :int64, "< 0")
973 cloopEmitTestSet(operands, :int, "< 0")
976 cloopEmitTestSet(operands, :int8, "== 0")
978 cloopEmitTestSet(operands, :int32, "== 0")
980 cloopEmitTestSet(operands, :int64, "== 0")
982 cloopEmitTestSet(operands, :int, "== 0")
985 cloopEmitTestSet(operands, :int8, "!
= 0")
987 cloopEmitTestSet(operands, :int32, "!
= 0")
989 cloopEmitTestSet(operands, :int64, "!
= 0")
991 cloopEmitTestSet(operands, :int, "!
= 0")
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.
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();"
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.
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.
1016 # Divide t1,t0 (EDX,EAX) by the specified arg, and store the remainder in t1,
1017 # and quotient in t0:
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();"
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.
1030 $asm.putc "#{operands[2].clValue(:double)} = Ints2Double(#{operands[0].clValue(:uint32)}, #{operands[1].clValue(:uint32)});"
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).
1035 $asm.putc "Double2Ints(#{operands[0].clValue(:double)}, #{operands[1].clValue(:uint32)}, #{operands[2].clValue(:uint32)});"
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.
1040 $asm.putc "#{operands[1].clValue(:double)} = #{operands[0].clValue(:castToDouble)};"
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.
1045 $asm.putc "#{operands[1].clValue(:int64)} = #{operands[0].clValue(:castToInt64)};"
1048 operands[0].cloopEmitLea(operands[1], :int32)
1050 operands[0].cloopEmitLea(operands[1], :int)
1053 cloopEmitOpAndBranchIfOverflow(operands, "+
", :int32)
1055 cloopEmitOpAndBranchIfOverflow(operands, "-", :int32)
1057 cloopEmitOpAndBranchIfOverflow(operands, "*", :int32)
1060 cloopEmitOpAndBranch(operands, "+
", :int32, "< 0")
1062 cloopEmitOpAndBranch(operands, "+
", :int32, "== 0")
1064 cloopEmitOpAndBranch(operands, "+
", :int32, "!
= 0")
1067 cloopEmitOpAndBranch(operands, "+
", :int64, "< 0")
1069 cloopEmitOpAndBranch(operands, "+
", :int64, "== 0")
1071 cloopEmitOpAndBranch(operands, "+
", :int64, "!
= 0")
1074 cloopEmitOpAndBranch(operands, "+
", :int, "< 0")
1076 cloopEmitOpAndBranch(operands, "+
", :int, "== 0")
1078 cloopEmitOpAndBranch(operands, "+
", :int, "!
= 0")
1081 cloopEmitOpAndBranch(operands, "-", :int32, "< 0")
1083 cloopEmitOpAndBranch(operands, "-", :int32, "== 0")
1085 cloopEmitOpAndBranch(operands, "-", :int32, "!
= 0")
1088 cloopEmitOpAndBranch(operands, "|", :int32, "< 0")
1090 cloopEmitOpAndBranch(operands, "|", :int32, "== 0")
1092 cloopEmitOpAndBranch(operands, "|", :int32, "!
= 0")
1099 $asm.putc "PUSH(#{op.clDump});"
1104 $asm.putc "POP(#{op.clDump});"
1107 when "pushCalleeSaves
"
1108 when "popCalleeSaves
"
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.
1117 $asm.putc "CRASH();"
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)
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)"
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
)
1148 when "cloopCallSlowPathVoid"
1149 cloopEmitCallSlowPathVoid(operands
)
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.
1155 $asm.putc
"#{annotation}"