module GHC.CmmToAsm.X86.Ppr (
pprNatCmmDecl,
pprData,
pprInstr,
pprFormat,
pprImm,
pprDataItem,
)
where
#include "HsVersions.h"
import GHC.Prelude
import GHC.Platform
import GHC.Platform.Reg
import GHC.CmmToAsm.X86.Regs
import GHC.CmmToAsm.X86.Instr
import GHC.CmmToAsm.X86.Cond
import GHC.CmmToAsm.Config
import GHC.CmmToAsm.Format
import GHC.CmmToAsm.Types
import GHC.CmmToAsm.Utils
import GHC.CmmToAsm.Ppr
import GHC.Cmm hiding (topInfoTable)
import GHC.Cmm.Dataflow.Collections
import GHC.Cmm.Dataflow.Label
import GHC.Cmm.BlockId
import GHC.Cmm.CLabel
import GHC.Types.Basic (Alignment, mkAlignment, alignmentBytes)
import GHC.Types.Unique ( pprUniqueAlways )
import GHC.Data.FastString
import GHC.Utils.Outputable
import GHC.Utils.Panic
import Data.Word
import Data.Bits
pprProcAlignment :: NCGConfig -> SDoc
pprProcAlignment config = maybe empty (pprAlign platform . mkAlignment) (ncgProcAlignment config)
where
platform = ncgPlatform config
pprNatCmmDecl :: NCGConfig -> NatCmmDecl (Alignment, RawCmmStatics) Instr -> SDoc
pprNatCmmDecl config (CmmData section dats) =
pprSectionAlign config section $$ pprDatas config dats
pprNatCmmDecl config proc@(CmmProc top_info lbl _ (ListGraph blocks)) =
let platform = ncgPlatform config in
pprProcAlignment config $$
case topInfoTable proc of
Nothing ->
pprSectionAlign config (Section Text lbl) $$
pprProcAlignment config $$
pprProcLabel config lbl $$
pprLabel platform lbl $$
vcat (map (pprBasicBlock config top_info) blocks) $$
ppWhen (ncgDwarfEnabled config) (pprBlockEndLabel platform lbl $$ pprProcEndLabel platform lbl) $$
pprSizeDecl platform lbl
Just (CmmStaticsRaw info_lbl _) ->
pprSectionAlign config (Section Text info_lbl) $$
pprProcAlignment config $$
pprProcLabel config lbl $$
(if platformHasSubsectionsViaSymbols platform
then pdoc platform (mkDeadStripPreventer info_lbl) <> char ':'
else empty) $$
vcat (map (pprBasicBlock config top_info) blocks) $$
ppWhen (ncgDwarfEnabled config) (pprProcEndLabel platform info_lbl) $$
(if platformHasSubsectionsViaSymbols platform
then
text "\t.long "
<+> pdoc platform info_lbl
<+> char '-'
<+> pdoc platform (mkDeadStripPreventer info_lbl)
else empty) $$
pprSizeDecl platform info_lbl
pprProcLabel :: NCGConfig -> CLabel -> SDoc
pprProcLabel config lbl
| ncgExposeInternalSymbols config
, Just lbl' <- ppInternalProcLabel (ncgThisModule config) lbl
= lbl' <> char ':'
| otherwise
= empty
pprProcEndLabel :: Platform -> CLabel
-> SDoc
pprProcEndLabel platform lbl =
pdoc platform (mkAsmTempProcEndLabel lbl) <> char ':'
pprBlockEndLabel :: Platform -> CLabel
-> SDoc
pprBlockEndLabel platform lbl =
pdoc platform (mkAsmTempEndLabel lbl) <> char ':'
pprSizeDecl :: Platform -> CLabel -> SDoc
pprSizeDecl platform lbl
= if osElfTarget (platformOS platform)
then text "\t.size" <+> pdoc platform lbl <> ptext (sLit ", .-") <> pdoc platform lbl
else empty
pprBasicBlock :: NCGConfig -> LabelMap RawCmmStatics -> NatBasicBlock Instr -> SDoc
pprBasicBlock config info_env (BasicBlock blockid instrs)
= maybe_infotable $
pprLabel platform asmLbl $$
vcat (map (pprInstr platform) instrs) $$
ppWhen (ncgDwarfEnabled config) (
pprBlockEndLabel platform asmLbl
<> pprProcEndLabel platform asmLbl
)
where
asmLbl = blockLbl blockid
platform = ncgPlatform config
maybe_infotable c = case mapLookup blockid info_env of
Nothing -> c
Just (CmmStaticsRaw infoLbl info) ->
pprAlignForSection platform Text $$
infoTableLoc $$
vcat (map (pprData config) info) $$
pprLabel platform infoLbl $$
c $$
ppWhen (ncgDwarfEnabled config) (pdoc platform (mkAsmTempEndLabel infoLbl) <> char ':')
infoTableLoc = case instrs of
(l@LOCATION{} : _) -> pprInstr platform l
_other -> empty
pprDatas :: NCGConfig -> (Alignment, RawCmmStatics) -> SDoc
pprDatas config (_, CmmStaticsRaw alias [CmmStaticLit (CmmLabel lbl), CmmStaticLit ind, _, _])
| lbl == mkIndStaticInfoLabel
, let labelInd (CmmLabelOff l _) = Just l
labelInd (CmmLabel l) = Just l
labelInd _ = Nothing
, Just ind' <- labelInd ind
, alias `mayRedirectTo` ind'
= pprGloblDecl (ncgPlatform config) alias
$$ text ".equiv" <+> pdoc (ncgPlatform config) alias <> comma <> pdoc (ncgPlatform config) (CmmLabel ind')
pprDatas config (align, (CmmStaticsRaw lbl dats))
= vcat (pprAlign platform align : pprLabel platform lbl : map (pprData config) dats)
where
platform = ncgPlatform config
pprData :: NCGConfig -> CmmStatic -> SDoc
pprData _config (CmmString str) = pprString str
pprData _config (CmmFileEmbed path) = pprFileEmbed path
pprData config (CmmUninitialised bytes)
= let platform = ncgPlatform config
in if platformOS platform == OSDarwin
then text ".space " <> int bytes
else text ".skip " <> int bytes
pprData config (CmmStaticLit lit) = pprDataItem config lit
pprGloblDecl :: Platform -> CLabel -> SDoc
pprGloblDecl platform lbl
| not (externallyVisibleCLabel lbl) = empty
| otherwise = text ".globl " <> pdoc platform lbl
pprLabelType' :: Platform -> CLabel -> SDoc
pprLabelType' platform lbl =
if isCFunctionLabel lbl || functionOkInfoTable then
text "@function"
else
text "@object"
where
functionOkInfoTable = platformTablesNextToCode platform &&
isInfoTableLabel lbl && not (isConInfoTableLabel lbl)
pprTypeDecl :: Platform -> CLabel -> SDoc
pprTypeDecl platform lbl
= if osElfTarget (platformOS platform) && externallyVisibleCLabel lbl
then text ".type " <> pdoc platform lbl <> ptext (sLit ", ") <> pprLabelType' platform lbl
else empty
pprLabel :: Platform -> CLabel -> SDoc
pprLabel platform lbl =
pprGloblDecl platform lbl
$$ pprTypeDecl platform lbl
$$ (pdoc platform lbl <> char ':')
pprAlign :: Platform -> Alignment -> SDoc
pprAlign platform alignment
= text ".align " <> int (alignmentOn platform)
where
bytes = alignmentBytes alignment
alignmentOn platform = if platformOS platform == OSDarwin
then log2 bytes
else bytes
log2 :: Int -> Int
log2 1 = 0
log2 2 = 1
log2 4 = 2
log2 8 = 3
log2 n = 1 + log2 (n `quot` 2)
pprReg :: Platform -> Format -> Reg -> SDoc
pprReg platform f r
= case r of
RegReal (RealRegSingle i) ->
if target32Bit platform then ppr32_reg_no f i
else ppr64_reg_no f i
RegReal (RealRegPair _ _) -> panic "X86.Ppr: no reg pairs on this arch"
RegVirtual (VirtualRegI u) -> text "%vI_" <> pprUniqueAlways u
RegVirtual (VirtualRegHi u) -> text "%vHi_" <> pprUniqueAlways u
RegVirtual (VirtualRegF u) -> text "%vF_" <> pprUniqueAlways u
RegVirtual (VirtualRegD u) -> text "%vD_" <> pprUniqueAlways u
where
ppr32_reg_no :: Format -> Int -> SDoc
ppr32_reg_no II8 = ppr32_reg_byte
ppr32_reg_no II16 = ppr32_reg_word
ppr32_reg_no _ = ppr32_reg_long
ppr32_reg_byte i = ptext
(case i of {
0 -> sLit "%al"; 1 -> sLit "%bl";
2 -> sLit "%cl"; 3 -> sLit "%dl";
_ -> sLit $ "very naughty I386 byte register: " ++ show i
})
ppr32_reg_word i = ptext
(case i of {
0 -> sLit "%ax"; 1 -> sLit "%bx";
2 -> sLit "%cx"; 3 -> sLit "%dx";
4 -> sLit "%si"; 5 -> sLit "%di";
6 -> sLit "%bp"; 7 -> sLit "%sp";
_ -> sLit "very naughty I386 word register"
})
ppr32_reg_long i = ptext
(case i of {
0 -> sLit "%eax"; 1 -> sLit "%ebx";
2 -> sLit "%ecx"; 3 -> sLit "%edx";
4 -> sLit "%esi"; 5 -> sLit "%edi";
6 -> sLit "%ebp"; 7 -> sLit "%esp";
_ -> ppr_reg_float i
})
ppr64_reg_no :: Format -> Int -> SDoc
ppr64_reg_no II8 = ppr64_reg_byte
ppr64_reg_no II16 = ppr64_reg_word
ppr64_reg_no II32 = ppr64_reg_long
ppr64_reg_no _ = ppr64_reg_quad
ppr64_reg_byte i = ptext
(case i of {
0 -> sLit "%al"; 1 -> sLit "%bl";
2 -> sLit "%cl"; 3 -> sLit "%dl";
4 -> sLit "%sil"; 5 -> sLit "%dil";
6 -> sLit "%bpl"; 7 -> sLit "%spl";
8 -> sLit "%r8b"; 9 -> sLit "%r9b";
10 -> sLit "%r10b"; 11 -> sLit "%r11b";
12 -> sLit "%r12b"; 13 -> sLit "%r13b";
14 -> sLit "%r14b"; 15 -> sLit "%r15b";
_ -> sLit $ "very naughty x86_64 byte register: " ++ show i
})
ppr64_reg_word i = ptext
(case i of {
0 -> sLit "%ax"; 1 -> sLit "%bx";
2 -> sLit "%cx"; 3 -> sLit "%dx";
4 -> sLit "%si"; 5 -> sLit "%di";
6 -> sLit "%bp"; 7 -> sLit "%sp";
8 -> sLit "%r8w"; 9 -> sLit "%r9w";
10 -> sLit "%r10w"; 11 -> sLit "%r11w";
12 -> sLit "%r12w"; 13 -> sLit "%r13w";
14 -> sLit "%r14w"; 15 -> sLit "%r15w";
_ -> sLit "very naughty x86_64 word register"
})
ppr64_reg_long i = ptext
(case i of {
0 -> sLit "%eax"; 1 -> sLit "%ebx";
2 -> sLit "%ecx"; 3 -> sLit "%edx";
4 -> sLit "%esi"; 5 -> sLit "%edi";
6 -> sLit "%ebp"; 7 -> sLit "%esp";
8 -> sLit "%r8d"; 9 -> sLit "%r9d";
10 -> sLit "%r10d"; 11 -> sLit "%r11d";
12 -> sLit "%r12d"; 13 -> sLit "%r13d";
14 -> sLit "%r14d"; 15 -> sLit "%r15d";
_ -> sLit "very naughty x86_64 register"
})
ppr64_reg_quad i = ptext
(case i of {
0 -> sLit "%rax"; 1 -> sLit "%rbx";
2 -> sLit "%rcx"; 3 -> sLit "%rdx";
4 -> sLit "%rsi"; 5 -> sLit "%rdi";
6 -> sLit "%rbp"; 7 -> sLit "%rsp";
8 -> sLit "%r8"; 9 -> sLit "%r9";
10 -> sLit "%r10"; 11 -> sLit "%r11";
12 -> sLit "%r12"; 13 -> sLit "%r13";
14 -> sLit "%r14"; 15 -> sLit "%r15";
_ -> ppr_reg_float i
})
ppr_reg_float :: Int -> PtrString
ppr_reg_float i = case i of
16 -> sLit "%xmm0" ; 17 -> sLit "%xmm1"
18 -> sLit "%xmm2" ; 19 -> sLit "%xmm3"
20 -> sLit "%xmm4" ; 21 -> sLit "%xmm5"
22 -> sLit "%xmm6" ; 23 -> sLit "%xmm7"
24 -> sLit "%xmm8" ; 25 -> sLit "%xmm9"
26 -> sLit "%xmm10"; 27 -> sLit "%xmm11"
28 -> sLit "%xmm12"; 29 -> sLit "%xmm13"
30 -> sLit "%xmm14"; 31 -> sLit "%xmm15"
_ -> sLit "very naughty x86 register"
pprFormat :: Format -> SDoc
pprFormat x
= ptext (case x of
II8 -> sLit "b"
II16 -> sLit "w"
II32 -> sLit "l"
II64 -> sLit "q"
FF32 -> sLit "ss"
FF64 -> sLit "sd"
)
pprFormat_x87 :: Format -> SDoc
pprFormat_x87 x
= ptext $ case x of
FF32 -> sLit "s"
FF64 -> sLit "l"
_ -> panic "X86.Ppr.pprFormat_x87"
pprCond :: Cond -> SDoc
pprCond c
= ptext (case c of {
GEU -> sLit "ae"; LU -> sLit "b";
EQQ -> sLit "e"; GTT -> sLit "g";
GE -> sLit "ge"; GU -> sLit "a";
LTT -> sLit "l"; LE -> sLit "le";
LEU -> sLit "be"; NE -> sLit "ne";
NEG -> sLit "s"; POS -> sLit "ns";
CARRY -> sLit "c"; OFLO -> sLit "o";
PARITY -> sLit "p"; NOTPARITY -> sLit "np";
ALWAYS -> sLit "mp"})
pprImm :: Platform -> Imm -> SDoc
pprImm platform = \case
ImmInt i -> int i
ImmInteger i -> integer i
ImmCLbl l -> pdoc platform l
ImmIndex l i -> pdoc platform l <> char '+' <> int i
ImmLit s -> s
ImmFloat f -> float $ fromRational f
ImmDouble d -> double $ fromRational d
ImmConstantSum a b -> pprImm platform a <> char '+' <> pprImm platform b
ImmConstantDiff a b -> pprImm platform a <> char '-' <> lparen <> pprImm platform b <> rparen
pprAddr :: Platform -> AddrMode -> SDoc
pprAddr platform (ImmAddr imm off)
= let pp_imm = pprImm platform imm
in
if (off == 0) then
pp_imm
else if (off < 0) then
pp_imm <> int off
else
pp_imm <> char '+' <> int off
pprAddr platform (AddrBaseIndex base index displacement)
= let
pp_disp = ppr_disp displacement
pp_off p = pp_disp <> char '(' <> p <> char ')'
pp_reg r = pprReg platform (archWordFormat (target32Bit platform)) r
in
case (base, index) of
(EABaseNone, EAIndexNone) -> pp_disp
(EABaseReg b, EAIndexNone) -> pp_off (pp_reg b)
(EABaseRip, EAIndexNone) -> pp_off (text "%rip")
(EABaseNone, EAIndex r i) -> pp_off (comma <> pp_reg r <> comma <> int i)
(EABaseReg b, EAIndex r i) -> pp_off (pp_reg b <> comma <> pp_reg r
<> comma <> int i)
_ -> panic "X86.Ppr.pprAddr: no match"
where
ppr_disp (ImmInt 0) = empty
ppr_disp imm = pprImm platform imm
pprSectionAlign :: NCGConfig -> Section -> SDoc
pprSectionAlign _config (Section (OtherSection _) _) =
panic "X86.Ppr.pprSectionAlign: unknown section"
pprSectionAlign config sec@(Section seg _) =
pprSectionHeader config sec $$
pprAlignForSection (ncgPlatform config) seg
pprAlignForSection :: Platform -> SectionType -> SDoc
pprAlignForSection platform seg =
text ".align " <>
case platformOS platform of
OSDarwin
| target32Bit platform ->
case seg of
ReadOnlyData16 -> int 4
CString -> int 1
_ -> int 2
| otherwise ->
case seg of
ReadOnlyData16 -> int 4
CString -> int 1
_ -> int 3
_
| target32Bit platform ->
case seg of
Text -> text "4,0x90"
ReadOnlyData16 -> int 16
CString -> int 1
_ -> int 4
| otherwise ->
case seg of
ReadOnlyData16 -> int 16
CString -> int 1
_ -> int 8
pprDataItem :: NCGConfig -> CmmLit -> SDoc
pprDataItem config lit
= vcat (ppr_item (cmmTypeFormat $ cmmLitType platform lit) lit)
where
platform = ncgPlatform config
imm = litToImm lit
ppr_item II8 _ = [text "\t.byte\t" <> pprImm platform imm]
ppr_item II16 _ = [text "\t.word\t" <> pprImm platform imm]
ppr_item II32 _ = [text "\t.long\t" <> pprImm platform imm]
ppr_item FF32 _ = [text "\t.float\t" <> pprImm platform imm]
ppr_item FF64 _ = [text "\t.double\t" <> pprImm platform imm]
ppr_item II64 _
= case platformOS platform of
OSDarwin
| target32Bit platform ->
case lit of
CmmInt x _ ->
[text "\t.long\t"
<> int (fromIntegral (fromIntegral x :: Word32)),
text "\t.long\t"
<> int (fromIntegral
(fromIntegral (x `shiftR` 32) :: Word32))]
_ -> panic "X86.Ppr.ppr_item: no match for II64"
| otherwise ->
[text "\t.quad\t" <> pprImm platform imm]
_
| target32Bit platform ->
[text "\t.quad\t" <> pprImm platform imm]
| otherwise ->
case lit of
CmmLabelDiffOff _ _ _ _ ->
[text "\t.long\t" <> pprImm platform imm,
text "\t.long\t0"]
_ ->
[text "\t.quad\t" <> pprImm platform imm]
asmComment :: SDoc -> SDoc
asmComment c = whenPprDebug $ text "# " <> c
pprInstr :: Platform -> Instr -> SDoc
pprInstr platform i = case i of
COMMENT s
-> asmComment (ftext s)
LOCATION file line col _name
-> text "\t.loc " <> ppr file <+> ppr line <+> ppr col
DELTA d
-> asmComment $ text ("\tdelta = " ++ show d)
NEWBLOCK _
-> panic "pprInstr: NEWBLOCK"
UNWIND lbl d
-> asmComment (text "\tunwind = " <> pdoc platform d)
$$ pdoc platform lbl <> colon
LDATA _ _
-> panic "pprInstr: LDATA"
MOV format (OpImm (ImmInt 0)) dst@(OpReg _)
-> pprInstr platform (XOR format' dst dst)
where format' = case format of
II64 -> II32
_ -> format
MOV format src dst
-> pprFormatOpOp (sLit "mov") format src dst
CMOV cc format src dst
-> pprCondOpReg (sLit "cmov") format cc src dst
MOVZxL II32 src dst
-> pprFormatOpOp (sLit "mov") II32 src dst
MOVZxL formats src dst
-> pprFormatOpOpCoerce (sLit "movz") formats II32 src dst
MOVSxL formats src dst
-> pprFormatOpOpCoerce (sLit "movs") formats (archWordFormat (target32Bit platform)) src dst
LEA format (OpAddr (AddrBaseIndex (EABaseReg reg1) (EAIndex reg2 1) (ImmInt 0))) dst@(OpReg reg3)
| reg1 == reg3
-> pprFormatOpOp (sLit "add") format (OpReg reg2) dst
LEA format (OpAddr (AddrBaseIndex (EABaseReg reg1) (EAIndex reg2 1) (ImmInt 0))) dst@(OpReg reg3)
| reg2 == reg3
-> pprFormatOpOp (sLit "add") format (OpReg reg1) dst
LEA format (OpAddr (AddrBaseIndex (EABaseReg reg1) EAIndexNone displ)) dst@(OpReg reg3)
| reg1 == reg3
-> pprInstr platform (ADD format (OpImm displ) dst)
LEA format src dst
-> pprFormatOpOp (sLit "lea") format src dst
ADD format (OpImm (ImmInt (1))) dst
-> pprFormatOp (sLit "dec") format dst
ADD format (OpImm (ImmInt 1)) dst
-> pprFormatOp (sLit "inc") format dst
ADD format src dst
-> pprFormatOpOp (sLit "add") format src dst
ADC format src dst
-> pprFormatOpOp (sLit "adc") format src dst
SUB format src dst
-> pprFormatOpOp (sLit "sub") format src dst
SBB format src dst
-> pprFormatOpOp (sLit "sbb") format src dst
IMUL format op1 op2
-> pprFormatOpOp (sLit "imul") format op1 op2
ADD_CC format src dst
-> pprFormatOpOp (sLit "add") format src dst
SUB_CC format src dst
-> pprFormatOpOp (sLit "sub") format src dst
AND II64 src@(OpImm (ImmInteger mask)) dst
| 0 <= mask && mask < 0xffffffff
-> pprInstr platform (AND II32 src dst)
AND FF32 src dst
-> pprOpOp (sLit "andps") FF32 src dst
AND FF64 src dst
-> pprOpOp (sLit "andpd") FF64 src dst
AND format src dst
-> pprFormatOpOp (sLit "and") format src dst
OR format src dst
-> pprFormatOpOp (sLit "or") format src dst
XOR FF32 src dst
-> pprOpOp (sLit "xorps") FF32 src dst
XOR FF64 src dst
-> pprOpOp (sLit "xorpd") FF64 src dst
XOR format src dst
-> pprFormatOpOp (sLit "xor") format src dst
POPCNT format src dst
-> pprOpOp (sLit "popcnt") format src (OpReg dst)
LZCNT format src dst
-> pprOpOp (sLit "lzcnt") format src (OpReg dst)
TZCNT format src dst
-> pprOpOp (sLit "tzcnt") format src (OpReg dst)
BSF format src dst
-> pprOpOp (sLit "bsf") format src (OpReg dst)
BSR format src dst
-> pprOpOp (sLit "bsr") format src (OpReg dst)
PDEP format src mask dst
-> pprFormatOpOpReg (sLit "pdep") format src mask dst
PEXT format src mask dst
-> pprFormatOpOpReg (sLit "pext") format src mask dst
PREFETCH NTA format src
-> pprFormatOp_ (sLit "prefetchnta") format src
PREFETCH Lvl0 format src
-> pprFormatOp_ (sLit "prefetcht0") format src
PREFETCH Lvl1 format src
-> pprFormatOp_ (sLit "prefetcht1") format src
PREFETCH Lvl2 format src
-> pprFormatOp_ (sLit "prefetcht2") format src
NOT format op
-> pprFormatOp (sLit "not") format op
BSWAP format op
-> pprFormatOp (sLit "bswap") format (OpReg op)
NEGI format op
-> pprFormatOp (sLit "neg") format op
SHL format src dst
-> pprShift (sLit "shl") format src dst
SAR format src dst
-> pprShift (sLit "sar") format src dst
SHR format src dst
-> pprShift (sLit "shr") format src dst
BT format imm src
-> pprFormatImmOp (sLit "bt") format imm src
CMP format src dst
| isFloatFormat format -> pprFormatOpOp (sLit "ucomi") format src dst
| otherwise -> pprFormatOpOp (sLit "cmp") format src dst
TEST format src dst
-> pprFormatOpOp (sLit "test") format' src dst
where
format' = case (src,dst) of
(OpImm (ImmInteger mask), OpReg dstReg)
| 0 <= mask && mask < 128 -> minSizeOfReg platform dstReg
_ -> format
minSizeOfReg platform (RegReal (RealRegSingle i))
| target32Bit platform && i <= 3 = II8
| target32Bit platform && i <= 7 = II16
| not (target32Bit platform) && i <= 15 = II8
minSizeOfReg _ _ = format
PUSH format op
-> pprFormatOp (sLit "push") format op
POP format op
-> pprFormatOp (sLit "pop") format op
NOP
-> text "\tnop"
CLTD II8
-> text "\tcbtw"
CLTD II16
-> text "\tcwtd"
CLTD II32
-> text "\tcltd"
CLTD II64
-> text "\tcqto"
CLTD x
-> panic $ "pprInstr: CLTD " ++ show x
SETCC cond op
-> pprCondInstr (sLit "set") cond (pprOperand platform II8 op)
XCHG format src val
-> pprFormatOpReg (sLit "xchg") format src val
JXX cond blockid
-> pprCondInstr (sLit "j") cond (pdoc platform lab)
where lab = blockLbl blockid
JXX_GBL cond imm
-> pprCondInstr (sLit "j") cond (pprImm platform imm)
JMP (OpImm imm) _
-> text "\tjmp " <> pprImm platform imm
JMP op _
-> text "\tjmp *" <> pprOperand platform (archWordFormat (target32Bit platform)) op
JMP_TBL op _ _ _
-> pprInstr platform (JMP op [])
CALL (Left imm) _
-> text "\tcall " <> pprImm platform imm
CALL (Right reg) _
-> text "\tcall *" <> pprReg platform (archWordFormat (target32Bit platform)) reg
IDIV fmt op
-> pprFormatOp (sLit "idiv") fmt op
DIV fmt op
-> pprFormatOp (sLit "div") fmt op
IMUL2 fmt op
-> pprFormatOp (sLit "imul") fmt op
MUL format op1 op2
-> pprFormatOpOp (sLit "mul") format op1 op2
MUL2 format op
-> pprFormatOp (sLit "mul") format op
FDIV format op1 op2
-> pprFormatOpOp (sLit "div") format op1 op2
SQRT format op1 op2
-> pprFormatOpReg (sLit "sqrt") format op1 op2
CVTSS2SD from to
-> pprRegReg (sLit "cvtss2sd") from to
CVTSD2SS from to
-> pprRegReg (sLit "cvtsd2ss") from to
CVTTSS2SIQ fmt from to
-> pprFormatFormatOpReg (sLit "cvttss2si") FF32 fmt from to
CVTTSD2SIQ fmt from to
-> pprFormatFormatOpReg (sLit "cvttsd2si") FF64 fmt from to
CVTSI2SS fmt from to
-> pprFormatOpReg (sLit "cvtsi2ss") fmt from to
CVTSI2SD fmt from to
-> pprFormatOpReg (sLit "cvtsi2sd") fmt from to
FETCHGOT reg
-> vcat [ text "\tcall 1f",
hcat [ text "1:\tpopl\t", pprReg platform II32 reg ],
hcat [ text "\taddl\t$_GLOBAL_OFFSET_TABLE_+(.-1b), ",
pprReg platform II32 reg ]
]
FETCHPC reg
-> vcat [ text "\tcall 1f",
hcat [ text "1:\tpopl\t", pprReg platform II32 reg ]
]
g@(X87Store fmt addr)
-> pprX87 g (hcat [gtab, text "fstp", pprFormat_x87 fmt, gsp, pprAddr platform addr])
LOCK i
-> text "\tlock" $$ pprInstr platform i
MFENCE
-> text "\tmfence"
XADD format src dst
-> pprFormatOpOp (sLit "xadd") format src dst
CMPXCHG format src dst
-> pprFormatOpOp (sLit "cmpxchg") format src dst
where
gtab :: SDoc
gtab = char '\t'
gsp :: SDoc
gsp = char ' '
pprX87 :: Instr -> SDoc -> SDoc
pprX87 fake actual
= (char '#' <> pprX87Instr fake) $$ actual
pprX87Instr :: Instr -> SDoc
pprX87Instr (X87Store fmt dst) = pprFormatAddr (sLit "gst") fmt dst
pprX87Instr _ = panic "X86.Ppr.pprX87Instr: no match"
pprDollImm :: Imm -> SDoc
pprDollImm i = text "$" <> pprImm platform i
pprOperand :: Platform -> Format -> Operand -> SDoc
pprOperand platform f op = case op of
OpReg r -> pprReg platform f r
OpImm i -> pprDollImm i
OpAddr ea -> pprAddr platform ea
pprMnemonic_ :: PtrString -> SDoc
pprMnemonic_ name =
char '\t' <> ptext name <> space
pprMnemonic :: PtrString -> Format -> SDoc
pprMnemonic name format =
char '\t' <> ptext name <> pprFormat format <> space
pprFormatImmOp :: PtrString -> Format -> Imm -> Operand -> SDoc
pprFormatImmOp name format imm op1
= hcat [
pprMnemonic name format,
char '$',
pprImm platform imm,
comma,
pprOperand platform format op1
]
pprFormatOp_ :: PtrString -> Format -> Operand -> SDoc
pprFormatOp_ name format op1
= hcat [
pprMnemonic_ name ,
pprOperand platform format op1
]
pprFormatOp :: PtrString -> Format -> Operand -> SDoc
pprFormatOp name format op1
= hcat [
pprMnemonic name format,
pprOperand platform format op1
]
pprFormatOpOp :: PtrString -> Format -> Operand -> Operand -> SDoc
pprFormatOpOp name format op1 op2
= hcat [
pprMnemonic name format,
pprOperand platform format op1,
comma,
pprOperand platform format op2
]
pprOpOp :: PtrString -> Format -> Operand -> Operand -> SDoc
pprOpOp name format op1 op2
= hcat [
pprMnemonic_ name,
pprOperand platform format op1,
comma,
pprOperand platform format op2
]
pprRegReg :: PtrString -> Reg -> Reg -> SDoc
pprRegReg name reg1 reg2
= hcat [
pprMnemonic_ name,
pprReg platform (archWordFormat (target32Bit platform)) reg1,
comma,
pprReg platform (archWordFormat (target32Bit platform)) reg2
]
pprFormatOpReg :: PtrString -> Format -> Operand -> Reg -> SDoc
pprFormatOpReg name format op1 reg2
= hcat [
pprMnemonic name format,
pprOperand platform format op1,
comma,
pprReg platform (archWordFormat (target32Bit platform)) reg2
]
pprCondOpReg :: PtrString -> Format -> Cond -> Operand -> Reg -> SDoc
pprCondOpReg name format cond op1 reg2
= hcat [
char '\t',
ptext name,
pprCond cond,
space,
pprOperand platform format op1,
comma,
pprReg platform format reg2
]
pprFormatFormatOpReg :: PtrString -> Format -> Format -> Operand -> Reg -> SDoc
pprFormatFormatOpReg name format1 format2 op1 reg2
= hcat [
pprMnemonic name format2,
pprOperand platform format1 op1,
comma,
pprReg platform format2 reg2
]
pprFormatOpOpReg :: PtrString -> Format -> Operand -> Operand -> Reg -> SDoc
pprFormatOpOpReg name format op1 op2 reg3
= hcat [
pprMnemonic name format,
pprOperand platform format op1,
comma,
pprOperand platform format op2,
comma,
pprReg platform format reg3
]
pprFormatAddr :: PtrString -> Format -> AddrMode -> SDoc
pprFormatAddr name format op
= hcat [
pprMnemonic name format,
comma,
pprAddr platform op
]
pprShift :: PtrString -> Format -> Operand -> Operand -> SDoc
pprShift name format src dest
= hcat [
pprMnemonic name format,
pprOperand platform II8 src,
comma,
pprOperand platform format dest
]
pprFormatOpOpCoerce :: PtrString -> Format -> Format -> Operand -> Operand -> SDoc
pprFormatOpOpCoerce name format1 format2 op1 op2
= hcat [ char '\t', ptext name, pprFormat format1, pprFormat format2, space,
pprOperand platform format1 op1,
comma,
pprOperand platform format2 op2
]
pprCondInstr :: PtrString -> Cond -> SDoc -> SDoc
pprCondInstr name cond arg
= hcat [ char '\t', ptext name, pprCond cond, space, arg]