1 # Copyright (C) 2012, 2014 Apple Inc. All rights reserved.
2 # Copyright (C) 2013 Digia Plc. and/or its subsidiary(-ies)
4 # Redistribution and use in source and binary forms, with or without
5 # modification, are permitted provided that the following conditions
7 # 1. Redistributions of source code must retain the above copyright
8 # notice, this list of conditions and the following disclaimer.
9 # 2. Redistributions in binary form must reproduce the above copyright
10 # notice, this list of conditions and the following disclaimer in the
11 # documentation and/or other materials provided with the distribution.
13 # THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14 # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15 # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 # PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17 # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23 # THE POSSIBILITY OF SUCH DAMAGE.
38 raise "bad value for $activeBackend: #{$activeBackend}"
53 raise "bad value for $activeBackend: #{$activeBackend}"
58 RUBY_PLATFORM =~
/cygwin/i
74 isIntelSyntax
? name
: "%" + name
77 def offsetRegister(off
, register
)
78 isIntelSyntax
? "[#{off} + #{register}]" : "#{off}(#{register})"
82 isIntelSyntax ? "" : "*"
85 def orderOperands(opA, opB)
86 isIntelSyntax ? "#{opB}, #{opA}" : "#{opA}, #{opB}"
90 isIntelSyntax ? "#{c}" : "$
#{c}"
93 def getSizeString(kind)
107 size
= isX64
? "qword" : "dword"
113 raise "Invalid kind #{kind}"
116 return size +
" " +
"ptr" +
" ";
119 class SpecialRegister
< NoChildren
121 raise unless @name =~
/^r/
125 register(@name +
"w")
127 register(@name +
"d")
136 def x86CallOperand(kind
)
137 # Call operands are not allowed to be partial registers.
138 "#{callPrefix}#{x86Operand(:quad)}"
142 X64_SCRATCH_REGISTER
= SpecialRegister
.new("r11")
145 def supports8BitOnX86
147 when "t0", "a0", "r0", "t1", "a1", "r1", "t2", "t3", "t4", "t5"
149 when "cfr", "ttnr", "tmr"
160 when "t0", "a0", "r0"
169 isX64
? register("rax") : register("eax")
171 isX64
? register("rax") : raise
173 raise "Invalid kind #{kind} for name #{name}"
175 when "t1", "a1", "r1"
184 isX64
? register("rdx") : register("edx")
186 isX64
? register("rdx") : raise
199 isX64
? register("rcx") : register("ecx")
201 isX64
? register("rcx") : raise
214 isX64
? register("rbx") : register("ebx")
216 isX64
? register("rbx") : raise
229 isX64
? register("rdi") : register("edi")
231 isX64
? register("rdi") : raise
270 isX64
? register("rsp") : register("esp")
272 isX64
? register("rsp") : raise
285 isX64
? register("rsi") : register("esi")
287 isX64
? register("rsi") : raise
290 raise "Cannot use #{name} in 32-bit X86 at #{codeOriginString}" unless isX64
302 raise "Cannot use #{name} in 32-bit X86 at #{codeOriginString}" unless isX64
314 raise "Cannot use #{name} in 32-bit X86 at #{codeOriginString}" unless isX64
326 raise "Cannot use #{name} in 32-bit X86 at #{codeOriginString}" unless isX64
338 raise "Bad register #{name} for X86 at #{codeOriginString}"
341 def x86CallOperand(kind
)
342 isX64
? "#{callPrefix}#{x86Operand(:quad)}" : "#{callPrefix}#{x86Operand(:ptr)}"
348 raise unless kind
== :double
351 when "ft0", "fa0", "fr"
364 raise "Bad register #{name} for X86 at #{codeOriginString}"
367 def x87DefaultStackPosition
373 when "ft2", "ft3", "ft4", "ft5"
374 raise "Unimplemented register #{name} for X86 at #{codeOriginString}"
376 raise "Bad register #{name} for X86 at #{codeOriginString}"
379 def x87Operand(offset
)
381 raise unless offset
== 0 or offset
== 1
382 "#{register("st")}(#{x87DefaultStackPosition + offset})"
384 def x86CallOperand(kind
)
385 "#{callPrefix}#{x86Operand(kind)}"
390 def validX86Immediate
?
392 value
>= -0x80000000 and value
<= 0x7fffffff
400 def x86CallOperand(kind
)
406 def supports8BitOnX86
410 def x86AddressOperand(addressKind
)
411 "#{offsetRegister(offset.value, base.x86Operand(addressKind))}"
414 "#{getSizeString(kind)}#{x86AddressOperand(:ptr)}"
416 def x86CallOperand(kind
)
417 "#{callPrefix}#{x86Operand(kind)}"
422 def supports8BitOnX86
426 def x86AddressOperand(addressKind
)
428 "#{offset.value}(#{base.x86Operand(addressKind)}, #{index.x86Operand(addressKind)}, #{scale})"
430 "#{getSizeString(addressKind)}[#{offset.value} + #{base.x86Operand(addressKind)} +
#{index.x86Operand(addressKind)} * #{scale}]"
436 x86AddressOperand(:ptr)
438 "#{getSizeString(kind)}[#{offset.value} + #{base.x86Operand(:ptr)} +
#{index.x86Operand(:ptr)} * #{scale}]"
442 def x86CallOperand(kind)
443 "#{callPrefix}#{x86Operand(kind)}"
447 class AbsoluteAddress
448 def supports8BitOnX86
452 def x86AddressOperand(addressKind)
460 def x86CallOperand(kind)
461 "#{callPrefix}#{address.value}"
466 def x86CallOperand(kind)
471 class LocalLabelReference
475 def x86CallOperand(kind)
481 def getModifiedListX86_64
487 if node.is_a? Instruction
488 unless node.opcode == "move
"
490 newOperands = node.operands.map {
492 if operand.immediate? and not operand.validX86Immediate?
494 raise "Attempt to use scratch register twice at
#{operand.codeOriginString}"
496 newList << Instruction.new(operand.codeOrigin, "move
", [operand, X64_SCRATCH_REGISTER])
503 newNode = Instruction.new(node.codeOrigin, node.opcode, newOperands, node.annotation)
506 unless node.is_a? Label or
507 node.is_a? LocalLabel or
509 raise "Unexpected
#{node.inspect} at
#{node.codeOrigin}"
519 def getModifiedListX86_64_WIN
520 getModifiedListX86_64
525 @@floatingPointCompareImplicitOperand = isIntelSyntax ? "st(0), " : ""
527 def x86Operands(*kinds)
528 raise unless kinds.size == operands.size
532 i = isIntelSyntax ? (kinds.size - idx - 1) : idx
533 result << operands[i].x86Operand(kinds[i])
555 not useX87 ? "sd
" : raise
580 def handleX86OpWithNumOperands(opcode, kind, numOperands)
582 if operands[0] == operands[2]
583 $asm.puts "#{opcode} #{orderOperands(operands[1].x86Operand(kind), operands[2].x86Operand(kind))}"
584 elsif operands[1] == operands[2]
585 $asm.puts "#{opcode} #{orderOperands(operands[0].x86Operand(kind), operands[2].x86Operand(kind))}"
587 $asm.puts "mov
#{x86Suffix(kind)} #{orderOperands(operands[0].x86Operand(kind), operands[2].x86Operand(kind))}"
588 $asm.puts "#{opcode} #{orderOperands(operands[1].x86Operand(kind), operands[2].x86Operand(kind))}"
591 $asm.puts "#{opcode} #{orderOperands(operands[0].x86Operand(kind), operands[1].x86Operand(kind))}"
595 def handleX86Op(opcode, kind)
596 handleX86OpWithNumOperands(opcode, kind, operands.size)
599 def handleX86Shift(opcode, kind)
600 if operands[0].is_a? Immediate or operands[0] == RegisterID.forName(nil, "t2
")
601 $asm.puts "#{opcode} #{orderOperands(operands[0].x86Operand(:byte), operands[1].x86Operand(kind))}"
603 cx = RegisterID.forName(nil, "t2
")
604 $asm.puts "xchg
#{x86Suffix(:ptr)} #{operands[0].x86Operand(:ptr)}, #{cx.x86Operand(:ptr)}"
605 $asm.puts "#{opcode} #{orderOperands(register("cl"), operands[1].x86Operand(kind))}"
606 $asm.puts "xchg
#{x86Suffix(:ptr)} #{operands[0].x86Operand(:ptr)}, #{cx.x86Operand(:ptr)}"
610 def handleX86DoubleBranch(branchOpcode, mode)
612 handleX87Compare(mode)
616 $asm.puts "ucomisd
#{orderOperands(operands[1].x86Operand(:double), operands[0].x86Operand(:double))}"
618 $asm.puts "ucomisd
#{orderOperands(operands[0].x86Operand(:double), operands[1].x86Operand(:double))}"
623 $asm.puts "#{branchOpcode} #{operands[2].asmLabel}"
626 def handleX86IntCompare(opcodeSuffix, kind)
627 if operands[0].is_a? Immediate and operands[0].value == 0 and operands[1].is_a? RegisterID and (opcodeSuffix == "e
" or opcodeSuffix == "ne
")
628 $asm.puts "test
#{x86Suffix(kind)} #{orderOperands(operands[1].x86Operand(kind), operands[1].x86Operand(kind))}"
629 elsif operands[1].is_a? Immediate and operands[1].value == 0 and operands[0].is_a? RegisterID and (opcodeSuffix == "e
" or opcodeSuffix == "ne
")
630 $asm.puts "test
#{x86Suffix(kind)} #{orderOperands(operands[0].x86Operand(kind), operands[0].x86Operand(kind))}"
632 $asm.puts "cmp
#{x86Suffix(kind)} #{orderOperands(operands[1].x86Operand(kind), operands[0].x86Operand(kind))}"
636 def handleX86IntBranch(branchOpcode, kind)
637 handleX86IntCompare(branchOpcode[1..-1], kind)
638 $asm.puts "#{branchOpcode} #{operands[2].asmLabel}"
641 def handleX86Set(setOpcode, operand)
642 if operand.supports8BitOnX86
643 $asm.puts "#{setOpcode} #{operand.x86Operand(:byte)}"
645 $asm.puts
"movzbl #{orderOperands(operand.x86Operand(:byte), operand.x86Operand(:int))}"
647 $asm.puts
"movzx #{orderOperands(operand.x86Operand(:byte), operand.x86Operand(:int))}"
650 ax
= RegisterID
.new(nil, "t0")
651 $asm.puts
"xchg#{x86Suffix(:ptr)} #{operand.x86Operand(:ptr)}, #{ax.x86Operand(:ptr)}"
652 $asm.puts
"#{setOpcode} %al"
653 $asm.puts
"movzbl %al, %eax"
654 $asm.puts
"xchg#{x86Suffix(:ptr)} #{operand.x86Operand(:ptr)}, #{ax.x86Operand(:ptr)}"
658 def handleX86IntCompareSet(setOpcode
, kind
)
659 handleX86IntCompare(setOpcode
[3..-1], kind
)
660 handleX86Set(setOpcode
, operands
[2])
663 def handleX86Test(kind
)
667 mask
= Immediate
.new(codeOrigin
, -1)
671 raise "Expected 2 or 3 operands, but got #{operands.size} at #{codeOriginString}"
674 if mask
.is_a
? Immediate
and mask
.value
== -1
675 if value
.is_a
? RegisterID
676 $asm.puts
"test#{x86Suffix(kind)} #{value.x86Operand(kind)}, #{value.x86Operand(kind)}"
678 $asm.puts
"cmp#{x86Suffix(kind)} #{orderOperands(const(0), value.x86Operand(kind))}"
681 $asm.puts
"test#{x86Suffix(kind)} #{orderOperands(mask.x86Operand(kind), value.x86Operand(kind))}"
685 def handleX86BranchTest(branchOpcode
, kind
)
687 $asm.puts
"#{branchOpcode} #{operands.last.asmLabel}"
690 def handleX86SetTest(setOpcode
, kind
)
692 handleX86Set(setOpcode
, operands
.last
)
695 def handleX86OpBranch(opcode
, branchOpcode
, kind
)
696 handleX86OpWithNumOperands(opcode
, kind
, operands
.size
- 1)
699 jumpTarget
= operands
[3]
701 jumpTarget
= operands
[2]
705 $asm.puts
"#{branchOpcode} #{jumpTarget.asmLabel}"
708 def handleX86SubBranch(branchOpcode
, kind
)
709 if operands
.size
== 4 and operands
[1] == operands
[2]
710 $asm.puts
"neg#{x86Suffix(kind)} #{operands[2].x86Operand(kind)}"
711 $asm.puts
"add#{x86Suffix(kind)} #{orderOperands(operands[0].x86Operand(kind), operands[2].x86Operand(kind))}"
713 handleX86OpWithNumOperands("sub#{x86Suffix(kind)}", kind
, operands
.size
- 1)
717 jumpTarget
= operands
[3]
719 jumpTarget
= operands
[2]
723 $asm.puts
"#{branchOpcode} #{jumpTarget.asmLabel}"
726 def handleX86Add(kind
)
727 if operands
.size
== 3 and operands
[1] == operands
[2]
728 unless Immediate
.new(nil, 0) == operands
[0]
729 $asm.puts
"add#{x86Suffix(kind)} #{orderOperands(operands[0].x86Operand(kind), operands[2].x86Operand(kind))}"
731 elsif operands
.size
== 3 and operands
[0].is_a
? Immediate
732 raise unless operands
[1].is_a
? RegisterID
733 raise unless operands
[2].is_a
? RegisterID
734 if operands
[0].value
== 0
735 unless operands
[1] == operands
[2]
736 $asm.puts
"mov#{x86Suffix(kind)} #{orderOperands(operands[1].x86Operand(kind), operands[2].x86Operand(kind))}"
739 $asm.puts
"lea#{x86Suffix(kind)} #{orderOperands(offsetRegister(operands[0].value, operands[1].x86Operand(kind)), operands[2].x86Operand(kind))}"
741 elsif operands
.size
== 3 and operands
[0].is_a
? RegisterID
742 raise unless operands
[1].is_a
? RegisterID
743 raise unless operands
[2].is_a
? RegisterID
744 if operands
[0] == operands
[2]
745 $asm.puts
"add#{x86Suffix(kind)} #{orderOperands(operands[1].x86Operand(kind), operands[2].x86Operand(kind))}"
748 $asm.puts
"lea#{x86Suffix(kind)} (#{operands[0].x86Operand(kind)}, #{operands[1].x86Operand(kind)}), #{operands[2].x86Operand(kind)}"
750 $asm.puts
"lea#{x86Suffix(kind)} #{operands[2].x86Operand(kind)}, [#{operands[0].x86Operand(kind)} + #{operands[1].x86Operand(kind)}]"
754 unless Immediate.new(nil, 0) == operands[0]
755 $asm.puts "add
#{x86Suffix(kind)} #{x86Operands(kind, kind)}"
760 def handleX86Sub(kind)
761 if operands.size == 3 and operands[1] == operands[2]
762 $asm.puts "neg
#{x86Suffix(kind)} #{operands[2].x86Operand(kind)}"
763 $asm.puts "add
#{x86Suffix(kind)} #{orderOperands(operands[0].x86Operand(kind), operands[2].x86Operand(kind))}"
765 handleX86Op("sub
#{x86Suffix(kind)}", kind)
769 def handleX86Mul(kind)
770 if operands.size == 3 and operands[0].is_a? Immediate
771 $asm.puts "imul
#{x86Suffix(kind)} #{x86Operands(kind, kind, kind)}"
773 # FIXME: could do some peephole in case the left operand is immediate and it's
775 handleX86Op("imul
#{x86Suffix(kind)}", kind)
780 sp = RegisterID.new(nil, "sp
")
781 opA = offsetRegister(operands[0].value * x86Bytes(:ptr), sp.x86Operand(:ptr))
782 opB = operands[1].x86Operand(:ptr)
783 $asm.puts "mov
#{x86Suffix(:ptr)} #{orderOperands(opA, opB)}"
787 sp = RegisterID.new(nil, "sp
")
788 opA = operands[0].x86Operand(:ptr)
789 opB = offsetRegister(operands[1].value * x86Bytes(:ptr), sp.x86Operand(:ptr))
790 $asm.puts "mov
#{x86Suffix(:ptr)} #{orderOperands(opA, opB)}"
794 if Immediate.new(nil, 0) == operands[0] and operands[1].is_a? RegisterID
796 $asm.puts "xor
#{x86Suffix(:quad)} #{operands[1].x86Operand(:quad)}, #{operands[1].x86Operand(:quad)}"
798 $asm.puts "xor
#{x86Suffix(:ptr)} #{operands[1].x86Operand(:ptr)}, #{operands[1].x86Operand(:ptr)}"
800 elsif operands[0] != operands
[1]
802 $asm.puts
"mov#{x86Suffix(:quad)} #{x86Operands(:quad, :quad)}"
804 $asm.puts
"mov#{x86Suffix(:ptr)} #{x86Operands(:ptr, :ptr)}"
809 def handleX87Compare(mode
)
812 if (operands
[0].x87DefaultStackPosition
== 0)
813 $asm.puts
"fucomi #{@@floatingPointCompareImplicitOperand}#{operands[1].x87Operand(0)}"
815 $asm.puts
"fld #{operands[0].x87Operand(0)}"
816 $asm.puts
"fucomip #{@@floatingPointCompareImplicitOperand}#{operands[1].x87Operand(1)}"
819 if (operands
[1].x87DefaultStackPosition
== 0)
820 $asm.puts
"fucomi #{@@floatingPointCompareImplicitOperand}#{operands[0].x87Operand(0)}"
822 $asm.puts
"fld #{operands[1].x87Operand(0)}"
823 $asm.puts
"fucomip #{@@floatingPointCompareImplicitOperand}#{operands[0].x87Operand(1)}"
830 def handleX87BinOp(opcode
, opcodereverse
)
831 if (operands
[1].x87DefaultStackPosition
== 0)
832 $asm.puts
"#{opcode} #{orderOperands(operands[0].x87Operand(0), register("st"))}"
833 elsif (operands
[0].x87DefaultStackPosition
== 0)
835 $asm.puts
"#{opcodereverse} #{register("st")}, #{operands[1].x87Operand(0)}"
837 $asm.puts
"#{opcode} #{operands[1].x87Operand(0)}, #{register("st")}"
840 $asm.puts
"fld #{operands[0].x87Operand(0)}"
841 $asm.puts
"#{opcodereverse}p #{orderOperands(register("st"), operands[1].x87Operand(1))}"
846 raise unless $activeBackend == "X86"
851 raise unless $activeBackend == "X86_WIN"
856 raise unless $activeBackend == "X86_64"
861 raise unless $activeBackend == "X86_64_WIN"
866 $asm.codeOrigin codeOriginString
if $enableCodeOriginComments
867 $asm.annotation annotation
if $enableInstrAnnotations
877 handleX86Op("and#{x86Suffix(:int)}", :int)
879 handleX86Op("and#{x86Suffix(:ptr)}", :ptr)
881 handleX86Op("and#{x86Suffix(:quad)}", :quad)
883 handleX86Shift("sal#{x86Suffix(:int)}", :int)
885 handleX86Shift("sal#{x86Suffix(:ptr)}", :ptr)
887 handleX86Shift("sal#{x86Suffix(:quad)}", :quad)
895 $asm.puts
"neg#{x86Suffix(:int)} #{x86Operands(:int)}"
897 $asm.puts
"neg#{x86Suffix(:ptr)} #{x86Operands(:ptr)}"
899 $asm.puts
"neg#{x86Suffix(:quad)} #{x86Operands(:quad)}"
901 $asm.puts
"not#{x86Suffix(:int)} #{x86Operands(:int)}"
903 handleX86Op("or#{x86Suffix(:int)}", :int)
905 handleX86Op("or#{x86Suffix(:ptr)}", :ptr)
907 handleX86Op("or#{x86Suffix(:quad)}", :quad)
909 handleX86Shift("sar#{x86Suffix(:int)}", :int)
911 handleX86Shift("sar#{x86Suffix(:ptr)}", :ptr)
913 handleX86Shift("sar#{x86Suffix(:quad)}", :quad)
915 handleX86Shift("shr#{x86Suffix(:int)}", :int)
917 handleX86Shift("shr#{x86Suffix(:ptr)}", :ptr)
919 handleX86Shift("shr#{x86Suffix(:quad)}", :quad)
927 handleX86Op("xor#{x86Suffix(:int)}", :int)
929 handleX86Op("xor#{x86Suffix(:ptr)}", :ptr)
931 handleX86Op("xor#{x86Suffix(:quad)}", :quad)
932 when "loadi", "storei"
933 $asm.puts
"mov#{x86Suffix(:int)} #{x86Operands(:int, :int)}"
937 $asm.puts
"movslq #{x86Operands(:int, :quad)}"
939 $asm.puts
"movsxd #{x86Operands(:int, :quad)}"
942 $asm.puts
"mov#{x86Suffix(:int)} #{x86Operands(:int, :int)}"
944 when "loadp", "storep"
945 $asm.puts
"mov#{x86Suffix(:ptr)} #{x86Operands(:ptr, :ptr)}"
946 when "loadq", "storeq"
947 $asm.puts
"mov#{x86Suffix(:quad)} #{x86Operands(:quad, :quad)}"
950 $asm.puts
"movzbl #{orderOperands(operands[0].x86Operand(:byte), operands[1].x86Operand(:int))}"
952 $asm.puts
"movzx #{orderOperands(operands[0].x86Operand(:byte), operands[1].x86Operand(:int))}"
955 $asm.puts
"movsbl #{operands[0].x86Operand(:byte)}, #{operands[1].x86Operand(:int)}"
958 $asm.puts
"movzwl #{orderOperands(operands[0].x86Operand(:half), operands[1].x86Operand(:int))}"
960 $asm.puts
"movzx #{orderOperands(operands[0].x86Operand(:half), operands[1].x86Operand(:int))}"
963 $asm.puts
"movswl #{operands[0].x86Operand(:half)}, #{operands[1].x86Operand(:int)}"
965 $asm.puts
"mov#{x86Suffix(:byte)} #{x86Operands(:byte, :byte)}"
969 $asm.puts
"fldl #{operands[0].x86Operand(:double)}"
971 $asm.puts
"fld #{operands[0].x86Operand(:double)}"
973 $asm.puts
"fstp #{operands[1].x87Operand(1)}"
975 $asm.puts
"movsd #{x86Operands(:double, :double)}"
979 if (operands
[0].x87DefaultStackPosition
== 0)
980 $asm.puts
"fst #{operands[1].x87Operand(0)}"
982 $asm.puts
"fld #{operands[0].x87Operand(0)}"
983 $asm.puts
"fstp #{operands[1].x87Operand(1)}"
986 $asm.puts
"movsd #{x86Operands(:double, :double)}"
990 if (operands
[0].x87DefaultStackPosition
== 0)
991 $asm.puts
"fst#{x86Suffix(:int)} #{operands[1].x86Operand(:double)}"
993 $asm.puts
"fld #{operands[0].x87Operand(0)}"
995 $asm.puts
"fstpl #{operands[1].x86Operand(:double)}"
997 $asm.puts
"fstp #{operands[1].x86Operand(:double)}"
1001 $asm.puts
"movsd #{x86Operands(:double, :double)}"
1005 handleX87BinOp("fadd", "fadd")
1007 $asm.puts
"addsd #{x86Operands(:double, :double)}"
1011 handleX87BinOp("fmul", "fmul")
1013 $asm.puts
"mulsd #{x86Operands(:double, :double)}"
1017 handleX87BinOp("fsub", "fsubr")
1019 $asm.puts
"subsd #{x86Operands(:double, :double)}"
1023 handleX87BinOp("fdiv", "fdivr")
1025 $asm.puts
"divsd #{x86Operands(:double, :double)}"
1029 $asm.puts
"fld #{operands[0].x87Operand(0)}"
1031 $asm.puts
"fstp #{operands[1].x87Operand(1)}"
1033 $asm.puts
"sqrtsd #{operands[0].x86Operand(:double)}, #{operands[1].x86Operand(:double)}"
1037 sp
= RegisterID
.new(nil, "sp")
1038 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(operands[0].x86Operand(:int), offsetRegister(-4, sp.x86Operand(:ptr)))}"
1039 $asm.puts
"fild#{x86Suffix(:ptr)} #{getSizeString(:ptr)}#{offsetRegister(-4, sp.x86Operand(:ptr))}"
1040 $asm.puts
"fstp #{operands[1].x87Operand(1)}"
1042 $asm.puts
"cvtsi2sd #{orderOperands(operands[0].x86Operand(:int), operands[1].x86Operand(:double))}"
1046 handleX87Compare(:normal)
1048 $asm.puts
"ucomisd #{orderOperands(operands[0].x86Operand(:double), operands[1].x86Operand(:double))}"
1050 if operands
[0] == operands
[1]
1051 # This is just a jump ordered, which is a jnp.
1052 $asm.puts
"jnp #{operands[2].asmLabel}"
1054 isUnordered
= LocalLabel
.unique("bdeq")
1055 $asm.puts
"jp #{LabelReference.new(codeOrigin, isUnordered).asmLabel}"
1056 $asm.puts
"je #{LabelReference.new(codeOrigin, operands[2]).asmLabel}"
1057 isUnordered
.lower("X86")
1060 handleX86DoubleBranch("jne", :normal)
1062 handleX86DoubleBranch("ja", :normal)
1064 handleX86DoubleBranch("jae", :normal)
1066 handleX86DoubleBranch("ja", :reverse)
1068 handleX86DoubleBranch("jae", :reverse)
1070 handleX86DoubleBranch("je", :normal)
1073 handleX87Compare(:normal)
1075 $asm.puts
"ucomisd #{orderOperands(operands[0].x86Operand(:double), operands[1].x86Operand(:double))}"
1077 if operands
[0] == operands
[1]
1078 # This is just a jump unordered, which is a jp.
1079 $asm.puts
"jp #{operands[2].asmLabel}"
1081 isUnordered
= LocalLabel
.unique("bdnequn")
1082 isEqual
= LocalLabel
.unique("bdnequn")
1083 $asm.puts
"jp #{LabelReference.new(codeOrigin, isUnordered).asmLabel}"
1084 $asm.puts
"je #{LabelReference.new(codeOrigin, isEqual).asmLabel}"
1085 isUnordered
.lower("X86")
1086 $asm.puts
"jmp #{operands[2].asmLabel}"
1087 isEqual
.lower("X86")
1090 handleX86DoubleBranch("jb", :reverse)
1092 handleX86DoubleBranch("jbe", :reverse)
1094 handleX86DoubleBranch("jb", :normal)
1096 handleX86DoubleBranch("jbe", :normal)
1098 # FIXME: unused and unimplemented for x87
1100 $asm.puts
"cvttsd2si #{operands[0].x86Operand(:double)}, #{operands[1].x86Operand(:int)}"
1101 $asm.puts
"cmpl $0x80000000 #{operands[1].x86Operand(:int)}"
1102 $asm.puts
"je #{operands[2].asmLabel}"
1104 # FIXME: unused and unimplemented for x87
1106 $asm.puts
"cvttsd2si #{operands[0].x86Operand(:double)}, #{operands[1].x86Operand(:int)}"
1109 sp
= RegisterID
.new(nil, "sp")
1110 if (operands
[0].x87DefaultStackPosition
== 0)
1111 $asm.puts
"fistl -4(#{sp.x86Operand(:ptr)})"
1113 $asm.puts
"fld #{operands[0].x87Operand(0)}"
1114 $asm.puts
"fistp#{x86Suffix(:ptr)} #{getSizeString(:ptr)}#{offsetRegister(-4, sp.x86Operand(:ptr))}"
1116 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(offsetRegister(-4, sp.x86Operand(:ptr)), operands[1].x86Operand(:int))}"
1117 $asm.puts
"test#{x86Suffix(:int)} #{operands[1].x86Operand(:int)}, #{operands[1].x86Operand(:int)}"
1118 $asm.puts
"je #{operands[2].asmLabel}"
1119 $asm.puts
"fild#{x86Suffix(:int)} #{getSizeString(:int)}#{offsetRegister(-4, sp.x86Operand(:ptr))}"
1120 $asm.puts
"fucomip #{@@floatingPointCompareImplicitOperand}#{operands[0].x87Operand(1)}"
1121 $asm.puts
"jp #{operands[2].asmLabel}"
1122 $asm.puts
"jne #{operands[2].asmLabel}"
1124 $asm.puts
"cvttsd2si #{operands[0].x86Operand(:double)}, #{operands[1].x86Operand(:int)}"
1125 $asm.puts
"test#{x86Suffix(:int)} #{operands[1].x86Operand(:int)}, #{operands[1].x86Operand(:int)}"
1126 $asm.puts
"je #{operands[2].asmLabel}"
1127 $asm.puts
"cvtsi2sd #{operands[1].x86Operand(:int)}, %xmm7"
1128 $asm.puts
"ucomisd #{operands[0].x86Operand(:double)}, %xmm7"
1129 $asm.puts
"jp #{operands[2].asmLabel}"
1130 $asm.puts
"jne #{operands[2].asmLabel}"
1135 $asm.puts
"fstp #{operands[0].x87Operand(1)}"
1137 $asm.puts
"xorpd #{operands[0].x86Operand(:double)}, #{operands[0].x86Operand(:double)}"
1142 $asm.puts
"pop #{op.x86Operand(:ptr)}"
1147 $asm.puts
"push #{op.x86Operand(:ptr)}"
1149 when "popCalleeSaves"
1152 $asm.puts
"pop " +
register("rsi")
1153 $asm.puts
"pop " +
register("rdi")
1155 $asm.puts
"pop " +
register("rbx")
1156 $asm.puts
"pop " +
register("r15")
1157 $asm.puts
"pop " +
register("r14")
1158 $asm.puts
"pop " +
register("r13")
1159 $asm.puts
"pop " +
register("r12")
1161 $asm.puts
"pop " +
register("ebx")
1162 $asm.puts
"pop " +
register("edi")
1163 $asm.puts
"pop " +
register("esi")
1165 when "pushCalleeSaves"
1167 $asm.puts
"push " +
register("r12")
1168 $asm.puts
"push " +
register("r13")
1169 $asm.puts
"push " +
register("r14")
1170 $asm.puts
"push " +
register("r15")
1171 $asm.puts
"push " +
register("rbx")
1173 $asm.puts
"push " +
register("rdi")
1174 $asm.puts
"push " +
register("rsi")
1177 $asm.puts
"push " +
register("esi")
1178 $asm.puts
"push " +
register("edi")
1179 $asm.puts
"push " +
register("ebx")
1185 $asm.puts
"movslq #{operands[0].x86Operand(:int)}, #{operands[1].x86Operand(:quad)}"
1187 $asm.puts
"movsxd #{orderOperands(operands[0].x86Operand(:int), operands[1].x86Operand(:quad))}"
1190 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(operands[0].x86Operand(:int), operands[1].x86Operand(:int))}"
1194 handleX86IntBranch("je", :int)
1196 handleX86IntBranch("je", :ptr)
1198 handleX86IntBranch("je", :quad)
1200 handleX86IntBranch("jne", :int)
1202 handleX86IntBranch("jne", :ptr)
1204 handleX86IntBranch("jne", :quad)
1206 handleX86IntBranch("ja", :int)
1208 handleX86IntBranch("ja", :ptr)
1210 handleX86IntBranch("ja", :quad)
1212 handleX86IntBranch("jae", :int)
1214 handleX86IntBranch("jae", :ptr)
1216 handleX86IntBranch("jae", :quad)
1218 handleX86IntBranch("jb", :int)
1220 handleX86IntBranch("jb", :ptr)
1222 handleX86IntBranch("jb", :quad)
1224 handleX86IntBranch("jbe", :int)
1226 handleX86IntBranch("jbe", :ptr)
1228 handleX86IntBranch("jbe", :quad)
1230 handleX86IntBranch("jg", :int)
1232 handleX86IntBranch("jg", :ptr)
1234 handleX86IntBranch("jg", :quad)
1236 handleX86IntBranch("jge", :int)
1238 handleX86IntBranch("jge", :ptr)
1240 handleX86IntBranch("jge", :quad)
1242 handleX86IntBranch("jl", :int)
1244 handleX86IntBranch("jl", :ptr)
1246 handleX86IntBranch("jl", :quad)
1248 handleX86IntBranch("jle", :int)
1250 handleX86IntBranch("jle", :ptr)
1252 handleX86IntBranch("jle", :quad)
1254 handleX86IntBranch("je", :byte)
1256 handleX86IntBranch("jne", :byte)
1258 handleX86IntBranch("ja", :byte)
1260 handleX86IntBranch("jae", :byte)
1262 handleX86IntBranch("jb", :byte)
1264 handleX86IntBranch("jbe", :byte)
1266 handleX86IntBranch("jg", :byte)
1268 handleX86IntBranch("jge", :byte)
1270 handleX86IntBranch("jl", :byte)
1272 handleX86IntBranch("jlteq", :byte)
1274 handleX86BranchTest("js", :int)
1276 handleX86BranchTest("js", :ptr)
1278 handleX86BranchTest("js", :quad)
1280 handleX86BranchTest("jz", :int)
1282 handleX86BranchTest("jz", :ptr)
1284 handleX86BranchTest("jz", :quad)
1286 handleX86BranchTest("jnz", :int)
1288 handleX86BranchTest("jnz", :ptr)
1290 handleX86BranchTest("jnz", :quad)
1292 handleX86BranchTest("js", :byte)
1294 handleX86BranchTest("jz", :byte)
1296 handleX86BranchTest("jnz", :byte)
1298 $asm.puts
"jmp #{operands[0].x86CallOperand(:ptr)}"
1300 handleX86OpBranch("add#{x86Suffix(:int)}", "jo", :int)
1302 handleX86OpBranch("add#{x86Suffix(:ptr)}", "jo", :ptr)
1304 handleX86OpBranch("add#{x86Suffix(:quad)}", "jo", :quad)
1306 handleX86OpBranch("add#{x86Suffix(:int)}", "js", :int)
1308 handleX86OpBranch("add#{x86Suffix(:ptr)}", "js", :ptr)
1310 handleX86OpBranch("add#{x86Suffix(:quad)}", "js", :quad)
1312 handleX86OpBranch("add#{x86Suffix(:int)}", "jz", :int)
1314 handleX86OpBranch("add#{x86Suffix(:ptr)}", "jz", :ptr)
1316 handleX86OpBranch("add#{x86Suffix(:quad)}", "jz", :quad)
1318 handleX86OpBranch("add#{x86Suffix(:int)}", "jnz", :int)
1320 handleX86OpBranch("add#{x86Suffix(:ptr)}", "jnz", :ptr)
1322 handleX86OpBranch("add#{x86Suffix(:quad)}", "jnz", :quad)
1324 handleX86SubBranch("jo", :int)
1326 handleX86SubBranch("js", :int)
1328 handleX86SubBranch("jz", :int)
1330 handleX86SubBranch("jnz", :int)
1332 handleX86OpBranch("imul#{x86Suffix(:int)}", "jo", :int)
1334 handleX86OpBranch("imul#{x86Suffix(:int)}", "js", :int)
1336 handleX86OpBranch("imul#{x86Suffix(:int)}", "jz", :int)
1338 handleX86OpBranch("imul#{x86Suffix(:int)}", "jnz", :int)
1340 handleX86OpBranch("orl", "jo", :int)
1342 handleX86OpBranch("orl", "js", :int)
1344 handleX86OpBranch("orl", "jz", :int)
1346 handleX86OpBranch("orl", "jnz", :int)
1348 $asm.puts
"int #{const(3)}"
1353 $asm.puts
"ffree #{register("st")}(#{offset})"
1356 op
= operands
[0].x86CallOperand(:ptr)
1357 if operands
[0].is_a
? LabelReference
1360 $asm.puts
"call #{op}"
1364 handleX86IntCompareSet("sete", :int)
1366 handleX86IntCompareSet("sete", :byte)
1368 handleX86IntCompareSet("sete", :ptr)
1370 handleX86IntCompareSet("sete", :quad)
1372 handleX86IntCompareSet("setne", :int)
1374 handleX86IntCompareSet("setne", :byte)
1376 handleX86IntCompareSet("setne", :ptr)
1378 handleX86IntCompareSet("setne", :quad)
1380 handleX86IntCompareSet("seta", :int)
1382 handleX86IntCompareSet("seta", :byte)
1384 handleX86IntCompareSet("seta", :ptr)
1386 handleX86IntCompareSet("seta", :quad)
1388 handleX86IntCompareSet("setae", :int)
1390 handleX86IntCompareSet("setae", :byte)
1392 handleX86IntCompareSet("setae", :ptr)
1394 handleX86IntCompareSet("setae", :quad)
1396 handleX86IntCompareSet("setb", :int)
1398 handleX86IntCompareSet("setb", :byte)
1400 handleX86IntCompareSet("setb", :ptr)
1402 handleX86IntCompareSet("setb", :quad)
1404 handleX86IntCompareSet("setbe", :int)
1406 handleX86IntCompareSet("setbe", :byte)
1408 handleX86IntCompareSet("setbe", :ptr)
1410 handleX86IntCompareSet("setbe", :quad)
1412 handleX86IntCompareSet("setg", :int)
1414 handleX86IntCompareSet("setg", :byte)
1416 handleX86IntCompareSet("setg", :ptr)
1418 handleX86IntCompareSet("setg", :quad)
1420 handleX86IntCompareSet("setge", :int)
1422 handleX86IntCompareSet("setge", :byte)
1424 handleX86IntCompareSet("setge", :ptr)
1426 handleX86IntCompareSet("setge", :quad)
1428 handleX86IntCompareSet("setl", :int)
1430 handleX86IntCompareSet("setl", :byte)
1432 handleX86IntCompareSet("setl", :ptr)
1434 handleX86IntCompareSet("setl", :quad)
1436 handleX86IntCompareSet("setle", :int)
1438 handleX86IntCompareSet("setle", :byte)
1440 handleX86IntCompareSet("setle", :ptr)
1442 handleX86IntCompareSet("setle", :quad)
1444 handleX86SetTest("sets", :int)
1446 handleX86SetTest("setz", :int)
1448 handleX86SetTest("setnz", :int)
1450 handleX86SetTest("sets", :ptr)
1452 handleX86SetTest("setz", :ptr)
1454 handleX86SetTest("setnz", :ptr)
1456 handleX86SetTest("sets", :quad)
1458 handleX86SetTest("setz", :quad)
1460 handleX86SetTest("setnz", :quad)
1462 handleX86SetTest("sets", :byte)
1464 handleX86SetTest("setz", :byte)
1466 handleX86SetTest("setnz", :byte)
1474 $asm.puts
"idiv#{x86Suffix(:int)} #{operands[0].x86Operand(:int)}"
1477 sp
= RegisterID
.new(nil, "sp")
1478 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(operands[0].x86Operand(:int), offsetRegister(-8, sp.x86Operand(:ptr)))}"
1479 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(operands[1].x86Operand(:int), offsetRegister(-4, sp.x86Operand(:ptr)))}"
1480 $asm.puts
"fld#{x86Suffix(:ptr)} #{getSizeString(:double)}#{offsetRegister(-8, sp.x86Operand(:ptr))}"
1481 $asm.puts
"fstp #{operands[2].x87Operand(1)}"
1483 $asm.puts
"movd #{operands[0].x86Operand(:int)}, #{operands[2].x86Operand(:double)}"
1484 $asm.puts
"movd #{operands[1].x86Operand(:int)}, %xmm7"
1485 $asm.puts
"psllq $32, %xmm7"
1486 $asm.puts
"por %xmm7, #{operands[2].x86Operand(:double)}"
1490 sp
= RegisterID
.new(nil, "sp")
1491 if (operands
[0].x87DefaultStackPosition
== 0)
1492 $asm.puts
"fst#{x86Suffix(:ptr)} #{getSizeString(:double)}#{offsetRegister(-8, sp.x86Operand(:ptr))}"
1494 $asm.puts
"fld #{operands[0].x87Operand(0)}"
1495 $asm.puts
"fstpl -8(#{sp.x86Operand(:ptr)})"
1497 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(offsetRegister(-8, sp.x86Operand(:ptr)), operands[1].x86Operand(:int))}"
1498 $asm.puts
"mov#{x86Suffix(:int)} #{orderOperands(offsetRegister(-4, sp.x86Operand(:ptr)), operands[2].x86Operand(:int))}"
1500 $asm.puts
"movd #{operands[0].x86Operand(:double)}, #{operands[1].x86Operand(:int)}"
1501 $asm.puts
"movsd #{operands[0].x86Operand(:double)}, %xmm7"
1502 $asm.puts
"psrlq $32, %xmm7"
1503 $asm.puts
"movd %xmm7, #{operands[2].x86Operand(:int)}"
1507 sp
= RegisterID
.new(nil, "sp")
1508 $asm.puts
"movq #{operands[0].x86Operand(:quad)}, -8(#{sp.x86Operand(:ptr)})"
1509 $asm.puts
"fldl -8(#{sp.x86Operand(:ptr)})"
1510 $asm.puts
"fstp #{operands[1].x87Operand(1)}"
1513 $asm.puts
"movq #{operands[0].x86Operand(:quad)}, #{operands[1].x86Operand(:double)}"
1515 # MASM does not accept register operands with movq.
1516 # Debugging shows that movd actually moves a qword when using MASM.
1517 $asm.puts
"movd #{operands[1].x86Operand(:double)}, #{operands[0].x86Operand(:quad)}"
1522 sp
= RegisterID
.new(nil, "sp")
1523 if (operands
[0].x87DefaultStackPosition
== 0)
1524 $asm.puts
"fst#{x86Suffix(:int)} #{getSizeString(:int)}#{offsetRegister(-8, sp.x86Operand(:ptr))}"
1526 $asm.puts
"fld #{operands[0].x87Operand(0)}"
1527 $asm.puts
"fstpl -8(#{sp.x86Operand(:ptr)})"
1529 $asm.puts
"movq -8(#{sp.x86Operand(:ptr)}), #{operands[1].x86Operand(:quad)}"
1532 $asm.puts
"movq #{operands[0].x86Operand(:double)}, #{operands[1].x86Operand(:quad)}"
1534 # MASM does not accept register operands with movq.
1535 # Debugging shows that movd actually moves a qword when using MASM.
1536 $asm.puts
"movd #{operands[1].x86Operand(:quad)}, #{operands[0].x86Operand(:double)}"
1540 $asm.puts
"jo #{operands[0].asmLabel}"
1542 $asm.puts
"js #{operands[0].asmLabel}"
1544 $asm.puts
"jz #{operands[0].asmLabel}"
1546 $asm.puts
"jnz #{operands[0].asmLabel}"
1548 $asm.puts
"lea#{x86Suffix(:int)} #{orderOperands(operands[0].x86AddressOperand(:int), operands[1].x86Operand(:int))}"
1550 $asm.puts
"lea#{x86Suffix(:ptr)} #{orderOperands(operands[0].x86AddressOperand(:ptr), operands[1].x86Operand(:ptr))}"