{-# LANGUAGE CPP, BangPatterns #-}
module StgCmmExpr ( cgExpr ) where
#include "HsVersions.h"
import GhcPrelude hiding ((<*>))
import {-# SOURCE #-} StgCmmBind ( cgBind )
import StgCmmMonad
import StgCmmHeap
import StgCmmEnv
import StgCmmCon
import StgCmmProf (saveCurrentCostCentre, restoreCurrentCostCentre, emitSetCCC)
import StgCmmLayout
import StgCmmPrim
import StgCmmHpc
import StgCmmTicky
import StgCmmUtils
import StgCmmClosure
import StgSyn
import MkGraph
import BlockId
import Cmm hiding ( succ )
import CmmInfo
import CoreSyn
import DataCon
import ForeignCall
import Id
import PrimOp
import TyCon
import Type ( isUnliftedType )
import RepType ( isVoidTy, countConRepArgs, primRepSlot )
import CostCentre ( CostCentreStack, currentCCS )
import Maybes
import Util
import FastString
import Outputable
import DynFlags
import Control.Monad ( unless, void )
import Control.Arrow ( first )
import Data.Function ( on )
import Data.List ( partition )
cgExpr :: CgStgExpr -> FCode ReturnKind
cgExpr :: CgStgExpr -> FCode ReturnKind
cgExpr (StgApp fun :: Id
fun args :: [StgArg]
args) = Id -> [StgArg] -> FCode ReturnKind
cgIdApp Id
fun [StgArg]
args
cgExpr (StgOpApp (StgPrimOp SeqOp) [StgVarArg a :: Id
a, _] _res_ty :: Type
_res_ty) =
Id -> [StgArg] -> FCode ReturnKind
cgIdApp Id
a []
cgExpr (StgOpApp (StgPrimOp DataToTagOp) [StgVarArg a :: Id
a] _res_ty :: Type
_res_ty) = do
DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
FastString -> FCode ()
emitComment (String -> FastString
mkFastString "dataToTag#")
LocalReg
tmp <- CmmType -> FCode LocalReg
forall (m :: * -> *). MonadUnique m => CmmType -> m LocalReg
newTemp (DynFlags -> CmmType
bWord DynFlags
dflags)
ReturnKind
_ <- Sequel -> FCode ReturnKind -> FCode ReturnKind
forall a. Sequel -> FCode a -> FCode a
withSequel ([LocalReg] -> Bool -> Sequel
AssignTo [LocalReg
tmp] Bool
False) (Id -> [StgArg] -> FCode ReturnKind
cgIdApp Id
a [])
[CmmExpr] -> FCode ReturnKind
emitReturn [DynFlags -> CmmExpr -> CmmExpr
getConstrTag DynFlags
dflags (DynFlags -> CmmExpr -> CmmExpr
cmmUntag DynFlags
dflags (CmmReg -> CmmExpr
CmmReg (LocalReg -> CmmReg
CmmLocal LocalReg
tmp)))]
cgExpr (StgOpApp op :: StgOp
op args :: [StgArg]
args ty :: Type
ty) = StgOp -> [StgArg] -> Type -> FCode ReturnKind
cgOpApp StgOp
op [StgArg]
args Type
ty
cgExpr (StgConApp con :: DataCon
con args :: [StgArg]
args _)= DataCon -> [StgArg] -> FCode ReturnKind
cgConApp DataCon
con [StgArg]
args
cgExpr (StgTick t :: Tickish Id
t e :: CgStgExpr
e) = Tickish Id -> FCode ()
cgTick Tickish Id
t FCode () -> FCode ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
e
cgExpr (StgLit lit :: Literal
lit) = do CmmLit
cmm_lit <- Literal -> FCode CmmLit
cgLit Literal
lit
[CmmExpr] -> FCode ReturnKind
emitReturn [CmmLit -> CmmExpr
CmmLit CmmLit
cmm_lit]
cgExpr (StgLet _ binds :: GenStgBinding 'CodeGen
binds expr :: CgStgExpr
expr) = do { GenStgBinding 'CodeGen -> FCode ()
cgBind GenStgBinding 'CodeGen
binds; CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
expr }
cgExpr (StgLetNoEscape _ binds :: GenStgBinding 'CodeGen
binds expr :: CgStgExpr
expr) =
do { Unique
u <- FCode Unique
newUnique
; let join_id :: BlockId
join_id = Unique -> BlockId
mkBlockId Unique
u
; BlockId -> GenStgBinding 'CodeGen -> FCode ()
cgLneBinds BlockId
join_id GenStgBinding 'CodeGen
binds
; ReturnKind
r <- CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
expr
; BlockId -> FCode ()
emitLabel BlockId
join_id
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
r }
cgExpr (StgCase expr :: CgStgExpr
expr bndr :: BinderP 'CodeGen
bndr alt_type :: AltType
alt_type alts :: [GenStgAlt 'CodeGen]
alts) =
CgStgExpr
-> Id -> AltType -> [GenStgAlt 'CodeGen] -> FCode ReturnKind
cgCase CgStgExpr
expr Id
BinderP 'CodeGen
bndr AltType
alt_type [GenStgAlt 'CodeGen]
alts
cgExpr (StgLam {}) = String -> FCode ReturnKind
forall a. String -> a
panic "cgExpr: StgLam"
cgLneBinds :: BlockId -> CgStgBinding -> FCode ()
cgLneBinds :: BlockId -> GenStgBinding 'CodeGen -> FCode ()
cgLneBinds join_id :: BlockId
join_id (StgNonRec bndr :: BinderP 'CodeGen
bndr rhs :: GenStgRhs 'CodeGen
rhs)
= do { Maybe LocalReg
local_cc <- FCode (Maybe LocalReg)
saveCurrentCostCentre
; (info :: CgIdInfo
info, fcode :: FCode ()
fcode) <- BlockId
-> Maybe LocalReg
-> Id
-> GenStgRhs 'CodeGen
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhs BlockId
join_id Maybe LocalReg
local_cc Id
BinderP 'CodeGen
bndr GenStgRhs 'CodeGen
rhs
; FCode ()
fcode
; CgIdInfo -> FCode ()
addBindC CgIdInfo
info }
cgLneBinds join_id :: BlockId
join_id (StgRec pairs :: [(BinderP 'CodeGen, GenStgRhs 'CodeGen)]
pairs)
= do { Maybe LocalReg
local_cc <- FCode (Maybe LocalReg)
saveCurrentCostCentre
; [(CgIdInfo, FCode ())]
r <- [FCode (CgIdInfo, FCode ())] -> FCode [(CgIdInfo, FCode ())]
forall (t :: * -> *) (m :: * -> *) a.
(Traversable t, Monad m) =>
t (m a) -> m (t a)
sequence ([FCode (CgIdInfo, FCode ())] -> FCode [(CgIdInfo, FCode ())])
-> [FCode (CgIdInfo, FCode ())] -> FCode [(CgIdInfo, FCode ())]
forall a b. (a -> b) -> a -> b
$ (Id -> GenStgRhs 'CodeGen -> FCode (CgIdInfo, FCode ()))
-> [(Id, GenStgRhs 'CodeGen)] -> [FCode (CgIdInfo, FCode ())]
forall a b c. (a -> b -> c) -> [(a, b)] -> [c]
unzipWith (BlockId
-> Maybe LocalReg
-> Id
-> GenStgRhs 'CodeGen
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhs BlockId
join_id Maybe LocalReg
local_cc) [(Id, GenStgRhs 'CodeGen)]
[(BinderP 'CodeGen, GenStgRhs 'CodeGen)]
pairs
; let (infos :: [CgIdInfo]
infos, fcodes :: [FCode ()]
fcodes) = [(CgIdInfo, FCode ())] -> ([CgIdInfo], [FCode ()])
forall a b. [(a, b)] -> ([a], [b])
unzip [(CgIdInfo, FCode ())]
r
; [CgIdInfo] -> FCode ()
addBindsC [CgIdInfo]
infos
; [FCode ()] -> FCode ()
forall (t :: * -> *) (m :: * -> *) a.
(Foldable t, Monad m) =>
t (m a) -> m ()
sequence_ [FCode ()]
fcodes
}
cgLetNoEscapeRhs
:: BlockId
-> Maybe LocalReg
-> Id
-> CgStgRhs
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhs :: BlockId
-> Maybe LocalReg
-> Id
-> GenStgRhs 'CodeGen
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhs join_id :: BlockId
join_id local_cc :: Maybe LocalReg
local_cc bndr :: Id
bndr rhs :: GenStgRhs 'CodeGen
rhs =
do { (info :: CgIdInfo
info, rhs_code :: FCode ()
rhs_code) <- Maybe LocalReg
-> Id -> GenStgRhs 'CodeGen -> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhsBody Maybe LocalReg
local_cc Id
bndr GenStgRhs 'CodeGen
rhs
; let (bid :: BlockId
bid, _) = String -> Maybe (BlockId, [LocalReg]) -> (BlockId, [LocalReg])
forall a. HasCallStack => String -> Maybe a -> a
expectJust "cgLetNoEscapeRhs" (Maybe (BlockId, [LocalReg]) -> (BlockId, [LocalReg]))
-> Maybe (BlockId, [LocalReg]) -> (BlockId, [LocalReg])
forall a b. (a -> b) -> a -> b
$ CgIdInfo -> Maybe (BlockId, [LocalReg])
maybeLetNoEscape CgIdInfo
info
; let code :: FCode ()
code = do { (_, body :: CmmAGraphScoped
body) <- FCode () -> FCode ((), CmmAGraphScoped)
forall a. FCode a -> FCode (a, CmmAGraphScoped)
getCodeScoped FCode ()
rhs_code
; BlockId -> CmmAGraphScoped -> FCode ()
emitOutOfLine BlockId
bid ((CmmAGraph -> CmmAGraph) -> CmmAGraphScoped -> CmmAGraphScoped
forall (a :: * -> * -> *) b c d.
Arrow a =>
a b c -> a (b, d) (c, d)
first (CmmAGraph -> CmmAGraph -> CmmAGraph
<*> BlockId -> CmmAGraph
mkBranch BlockId
join_id) CmmAGraphScoped
body) }
; (CgIdInfo, FCode ()) -> FCode (CgIdInfo, FCode ())
forall (m :: * -> *) a. Monad m => a -> m a
return (CgIdInfo
info, FCode ()
code)
}
cgLetNoEscapeRhsBody
:: Maybe LocalReg
-> Id
-> CgStgRhs
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhsBody :: Maybe LocalReg
-> Id -> GenStgRhs 'CodeGen -> FCode (CgIdInfo, FCode ())
cgLetNoEscapeRhsBody local_cc :: Maybe LocalReg
local_cc bndr :: Id
bndr (StgRhsClosure _ cc :: CostCentreStack
cc _upd :: UpdateFlag
_upd args :: [BinderP 'CodeGen]
args body :: CgStgExpr
body)
= Id
-> Maybe LocalReg
-> CostCentreStack
-> [NonVoid Id]
-> CgStgExpr
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeClosure Id
bndr Maybe LocalReg
local_cc CostCentreStack
cc ([Id] -> [NonVoid Id]
nonVoidIds [Id]
[BinderP 'CodeGen]
args) CgStgExpr
body
cgLetNoEscapeRhsBody local_cc :: Maybe LocalReg
local_cc bndr :: Id
bndr (StgRhsCon cc :: CostCentreStack
cc con :: DataCon
con args :: [StgArg]
args)
= Id
-> Maybe LocalReg
-> CostCentreStack
-> [NonVoid Id]
-> CgStgExpr
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeClosure Id
bndr Maybe LocalReg
local_cc CostCentreStack
cc []
(DataCon -> [StgArg] -> [Type] -> CgStgExpr
forall (pass :: StgPass).
DataCon -> [StgArg] -> [Type] -> GenStgExpr pass
StgConApp DataCon
con [StgArg]
args (String -> SDoc -> [Type]
forall a. HasCallStack => String -> SDoc -> a
pprPanic "cgLetNoEscapeRhsBody" (SDoc -> [Type]) -> SDoc -> [Type]
forall a b. (a -> b) -> a -> b
$
String -> SDoc
text "StgRhsCon doesn't have type args"))
cgLetNoEscapeClosure
:: Id
-> Maybe LocalReg
-> CostCentreStack
-> [NonVoid Id]
-> CgStgExpr
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeClosure :: Id
-> Maybe LocalReg
-> CostCentreStack
-> [NonVoid Id]
-> CgStgExpr
-> FCode (CgIdInfo, FCode ())
cgLetNoEscapeClosure bndr :: Id
bndr cc_slot :: Maybe LocalReg
cc_slot _unused_cc :: CostCentreStack
_unused_cc args :: [NonVoid Id]
args body :: CgStgExpr
body
= do DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
(CgIdInfo, FCode ()) -> FCode (CgIdInfo, FCode ())
forall (m :: * -> *) a. Monad m => a -> m a
return ( DynFlags -> Id -> [NonVoid Id] -> CgIdInfo
lneIdInfo DynFlags
dflags Id
bndr [NonVoid Id]
args
, FCode ()
code )
where
code :: FCode ()
code = FCode () -> FCode ()
forall a. FCode a -> FCode a
forkLneBody (FCode () -> FCode ()) -> FCode () -> FCode ()
forall a b. (a -> b) -> a -> b
$ do {
; Name -> [NonVoid Id] -> FCode () -> FCode ()
forall a. Name -> [NonVoid Id] -> FCode a -> FCode a
withNewTickyCounterLNE (Id -> Name
idName Id
bndr) [NonVoid Id]
args (FCode () -> FCode ()) -> FCode () -> FCode ()
forall a b. (a -> b) -> a -> b
$ do
; Maybe LocalReg -> FCode ()
restoreCurrentCostCentre Maybe LocalReg
cc_slot
; [LocalReg]
arg_regs <- [NonVoid Id] -> FCode [LocalReg]
bindArgsToRegs [NonVoid Id]
args
; FCode ReturnKind -> FCode ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (FCode ReturnKind -> FCode ()) -> FCode ReturnKind -> FCode ()
forall a b. (a -> b) -> a -> b
$ [LocalReg] -> FCode ReturnKind -> FCode ReturnKind
forall a. [LocalReg] -> FCode a -> FCode a
noEscapeHeapCheck [LocalReg]
arg_regs (FCode ()
tickyEnterLNE FCode () -> FCode ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
body) }
data GcPlan
= GcInAlts
[LocalReg]
| NoGcInAlts
cgCase :: CgStgExpr -> Id -> AltType -> [CgStgAlt] -> FCode ReturnKind
cgCase :: CgStgExpr
-> Id -> AltType -> [GenStgAlt 'CodeGen] -> FCode ReturnKind
cgCase (StgOpApp (StgPrimOp op :: PrimOp
op) args :: [StgArg]
args _) bndr :: Id
bndr (AlgAlt tycon :: TyCon
tycon) alts :: [GenStgAlt 'CodeGen]
alts
| TyCon -> Bool
isEnumerationTyCon TyCon
tycon
= do { CmmExpr
tag_expr <- PrimOp -> [StgArg] -> FCode CmmExpr
do_enum_primop PrimOp
op [StgArg]
args
; Bool -> FCode () -> FCode ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless (Id -> Bool
isDeadBinder Id
bndr) (FCode () -> FCode ()) -> FCode () -> FCode ()
forall a b. (a -> b) -> a -> b
$ do
{ DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; LocalReg
tmp_reg <- NonVoid Id -> FCode LocalReg
bindArgToReg (Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
bndr)
; CmmReg -> CmmExpr -> FCode ()
emitAssign (LocalReg -> CmmReg
CmmLocal LocalReg
tmp_reg)
(DynFlags -> TyCon -> CmmExpr -> CmmExpr
tagToClosure DynFlags
dflags TyCon
tycon CmmExpr
tag_expr) }
; (mb_deflt :: Maybe CmmAGraphScoped
mb_deflt, branches :: [(ConTagZ, CmmAGraphScoped)]
branches) <- (GcPlan, ReturnKind)
-> NonVoid Id
-> [GenStgAlt 'CodeGen]
-> FCode (Maybe CmmAGraphScoped, [(ConTagZ, CmmAGraphScoped)])
cgAlgAltRhss (GcPlan
NoGcInAlts,ReturnKind
AssignedDirectly)
(Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
bndr) [GenStgAlt 'CodeGen]
alts
; CmmExpr
-> [(ConTagZ, CmmAGraphScoped)]
-> Maybe CmmAGraphScoped
-> ConTagZ
-> ConTagZ
-> FCode ()
emitSwitch CmmExpr
tag_expr [(ConTagZ, CmmAGraphScoped)]
branches Maybe CmmAGraphScoped
mb_deflt 0 (TyCon -> ConTagZ
tyConFamilySize TyCon
tycon ConTagZ -> ConTagZ -> ConTagZ
forall a. Num a => a -> a -> a
- 1)
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
AssignedDirectly
}
where
do_enum_primop :: PrimOp -> [StgArg] -> FCode CmmExpr
do_enum_primop :: PrimOp -> [StgArg] -> FCode CmmExpr
do_enum_primop TagToEnumOp [arg :: StgArg
arg]
= NonVoid StgArg -> FCode CmmExpr
getArgAmode (StgArg -> NonVoid StgArg
forall a. a -> NonVoid a
NonVoid StgArg
arg)
do_enum_primop primop :: PrimOp
primop args :: [StgArg]
args
= do DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
LocalReg
tmp <- CmmType -> FCode LocalReg
forall (m :: * -> *). MonadUnique m => CmmType -> m LocalReg
newTemp (DynFlags -> CmmType
bWord DynFlags
dflags)
[LocalReg] -> PrimOp -> [StgArg] -> FCode ()
cgPrimOp [LocalReg
tmp] PrimOp
primop [StgArg]
args
CmmExpr -> FCode CmmExpr
forall (m :: * -> *) a. Monad m => a -> m a
return (CmmReg -> CmmExpr
CmmReg (LocalReg -> CmmReg
CmmLocal LocalReg
tmp))
cgCase (StgApp v :: Id
v []) _ (PrimAlt _) alts :: [GenStgAlt 'CodeGen]
alts
| PrimRep -> Bool
isVoidRep (Id -> PrimRep
idPrimRep Id
v)
, [(DEFAULT, _, rhs :: CgStgExpr
rhs)] <- [GenStgAlt 'CodeGen]
alts
= CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
rhs
cgCase (StgApp v :: Id
v []) bndr :: Id
bndr alt_type :: AltType
alt_type@(PrimAlt _) alts :: [GenStgAlt 'CodeGen]
alts
| HasDebugCallStack => Type -> Bool
Type -> Bool
isUnliftedType (Id -> Type
idType Id
v)
Bool -> Bool -> Bool
|| Bool
reps_compatible
=
do { DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; Bool -> FCode () -> FCode ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless Bool
reps_compatible (FCode () -> FCode ()) -> FCode () -> FCode ()
forall a b. (a -> b) -> a -> b
$
String -> SDoc -> FCode ()
forall a. HasCallStack => String -> SDoc -> a
pprPanic "cgCase: reps do not match, perhaps a dodgy unsafeCoerce?"
(Id -> SDoc
pp_bndr Id
v SDoc -> SDoc -> SDoc
$$ Id -> SDoc
pp_bndr Id
bndr)
; CgIdInfo
v_info <- Id -> FCode CgIdInfo
getCgIdInfo Id
v
; CmmReg -> CmmExpr -> FCode ()
emitAssign (LocalReg -> CmmReg
CmmLocal (DynFlags -> NonVoid Id -> LocalReg
idToReg DynFlags
dflags (Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
bndr)))
(CgIdInfo -> CmmExpr
idInfoToAmode CgIdInfo
v_info)
; LocalReg
_ <- NonVoid Id -> FCode LocalReg
bindArgToReg (Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
bndr)
; (GcPlan, ReturnKind)
-> NonVoid Id
-> AltType
-> [GenStgAlt 'CodeGen]
-> FCode ReturnKind
cgAlts (GcPlan
NoGcInAlts,ReturnKind
AssignedDirectly) (Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
bndr) AltType
alt_type [GenStgAlt 'CodeGen]
alts }
where
reps_compatible :: Bool
reps_compatible = (SlotTy -> SlotTy -> Bool
forall a. Eq a => a -> a -> Bool
(==) (SlotTy -> SlotTy -> Bool) -> (Id -> SlotTy) -> Id -> Id -> Bool
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` (PrimRep -> SlotTy
primRepSlot (PrimRep -> SlotTy) -> (Id -> PrimRep) -> Id -> SlotTy
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Id -> PrimRep
idPrimRep)) Id
v Id
bndr
pp_bndr :: Id -> SDoc
pp_bndr id :: Id
id = Id -> SDoc
forall a. Outputable a => a -> SDoc
ppr Id
id SDoc -> SDoc -> SDoc
<+> SDoc
dcolon SDoc -> SDoc -> SDoc
<+> Type -> SDoc
forall a. Outputable a => a -> SDoc
ppr (Id -> Type
idType Id
id) SDoc -> SDoc -> SDoc
<+> SDoc -> SDoc
parens (PrimRep -> SDoc
forall a. Outputable a => a -> SDoc
ppr (Id -> PrimRep
idPrimRep Id
id))
cgCase scrut :: CgStgExpr
scrut@(StgApp v :: Id
v []) _ (PrimAlt _) _
= do { DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; Maybe LocalReg
mb_cc <- Bool -> FCode (Maybe LocalReg)
maybeSaveCostCentre Bool
True
; ReturnKind
_ <- Sequel -> FCode ReturnKind -> FCode ReturnKind
forall a. Sequel -> FCode a -> FCode a
withSequel
([LocalReg] -> Bool -> Sequel
AssignTo [DynFlags -> NonVoid Id -> LocalReg
idToReg DynFlags
dflags (Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
v)] Bool
False) (CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
scrut)
; Maybe LocalReg -> FCode ()
restoreCurrentCostCentre Maybe LocalReg
mb_cc
; FastString -> FCode ()
emitComment (FastString -> FCode ()) -> FastString -> FCode ()
forall a b. (a -> b) -> a -> b
$ String -> FastString
mkFastString "should be unreachable code"
; BlockId
l <- FCode BlockId
forall (m :: * -> *). MonadUnique m => m BlockId
newBlockId
; BlockId -> FCode ()
emitLabel BlockId
l
; CmmAGraph -> FCode ()
emit (BlockId -> CmmAGraph
mkBranch BlockId
l)
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
AssignedDirectly
}
cgCase (StgOpApp (StgPrimOp SeqOp) [StgVarArg a :: Id
a, _] _) bndr :: Id
bndr alt_type :: AltType
alt_type alts :: [GenStgAlt 'CodeGen]
alts
=
CgStgExpr
-> Id -> AltType -> [GenStgAlt 'CodeGen] -> FCode ReturnKind
cgCase (Id -> [StgArg] -> CgStgExpr
forall (pass :: StgPass). Id -> [StgArg] -> GenStgExpr pass
StgApp Id
a []) Id
bndr AltType
alt_type [GenStgAlt 'CodeGen]
alts
cgCase scrut :: CgStgExpr
scrut bndr :: Id
bndr alt_type :: AltType
alt_type alts :: [GenStgAlt 'CodeGen]
alts
=
do { DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; ConTagZ
up_hp_usg <- FCode ConTagZ
getVirtHp
; let ret_bndrs :: [NonVoid Id]
ret_bndrs = Id -> AltType -> [GenStgAlt 'CodeGen] -> [NonVoid Id]
chooseReturnBndrs Id
bndr AltType
alt_type [GenStgAlt 'CodeGen]
alts
alt_regs :: [LocalReg]
alt_regs = (NonVoid Id -> LocalReg) -> [NonVoid Id] -> [LocalReg]
forall a b. (a -> b) -> [a] -> [b]
map (DynFlags -> NonVoid Id -> LocalReg
idToReg DynFlags
dflags) [NonVoid Id]
ret_bndrs
; Bool
simple_scrut <- CgStgExpr -> AltType -> FCode Bool
isSimpleScrut CgStgExpr
scrut AltType
alt_type
; let do_gc :: Bool
do_gc | CgStgExpr -> Bool
forall (pass :: StgPass). GenStgExpr pass -> Bool
is_cmp_op CgStgExpr
scrut = Bool
False
| Bool -> Bool
not Bool
simple_scrut = Bool
True
| [(AltCon, [Id], CgStgExpr)] -> Bool
forall a. [a] -> Bool
isSingleton [(AltCon, [Id], CgStgExpr)]
[GenStgAlt 'CodeGen]
alts = Bool
False
| ConTagZ
up_hp_usg ConTagZ -> ConTagZ -> Bool
forall a. Ord a => a -> a -> Bool
> 0 = Bool
False
| Bool
otherwise = Bool
True
gc_plan :: GcPlan
gc_plan = if Bool
do_gc then [LocalReg] -> GcPlan
GcInAlts [LocalReg]
alt_regs else GcPlan
NoGcInAlts
; Maybe LocalReg
mb_cc <- Bool -> FCode (Maybe LocalReg)
maybeSaveCostCentre Bool
simple_scrut
; let sequel :: Sequel
sequel = [LocalReg] -> Bool -> Sequel
AssignTo [LocalReg]
alt_regs Bool
do_gc
; ReturnKind
ret_kind <- Sequel -> FCode ReturnKind -> FCode ReturnKind
forall a. Sequel -> FCode a -> FCode a
withSequel Sequel
sequel (CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
scrut)
; Maybe LocalReg -> FCode ()
restoreCurrentCostCentre Maybe LocalReg
mb_cc
; [LocalReg]
_ <- [NonVoid Id] -> FCode [LocalReg]
bindArgsToRegs [NonVoid Id]
ret_bndrs
; (GcPlan, ReturnKind)
-> NonVoid Id
-> AltType
-> [GenStgAlt 'CodeGen]
-> FCode ReturnKind
cgAlts (GcPlan
gc_plan,ReturnKind
ret_kind) (Id -> NonVoid Id
forall a. a -> NonVoid a
NonVoid Id
bndr) AltType
alt_type [GenStgAlt 'CodeGen]
alts
}
where
is_cmp_op :: GenStgExpr pass -> Bool
is_cmp_op (StgOpApp (StgPrimOp op :: PrimOp
op) _ _) = PrimOp -> Bool
isComparisonPrimOp PrimOp
op
is_cmp_op _ = Bool
False
maybeSaveCostCentre :: Bool -> FCode (Maybe LocalReg)
maybeSaveCostCentre :: Bool -> FCode (Maybe LocalReg)
maybeSaveCostCentre simple_scrut :: Bool
simple_scrut
| Bool
simple_scrut = Maybe LocalReg -> FCode (Maybe LocalReg)
forall (m :: * -> *) a. Monad m => a -> m a
return Maybe LocalReg
forall a. Maybe a
Nothing
| Bool
otherwise = FCode (Maybe LocalReg)
saveCurrentCostCentre
isSimpleScrut :: CgStgExpr -> AltType -> FCode Bool
isSimpleScrut :: CgStgExpr -> AltType -> FCode Bool
isSimpleScrut (StgOpApp op :: StgOp
op args :: [StgArg]
args _) _ = StgOp -> [StgArg] -> FCode Bool
isSimpleOp StgOp
op [StgArg]
args
isSimpleScrut (StgLit _) _ = Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return Bool
True
isSimpleScrut (StgApp _ []) (PrimAlt _) = Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return Bool
True
isSimpleScrut _ _ = Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return Bool
False
isSimpleOp :: StgOp -> [StgArg] -> FCode Bool
isSimpleOp :: StgOp -> [StgArg] -> FCode Bool
isSimpleOp (StgFCallOp (CCall (CCallSpec _ _ safe :: Safety
safe)) _) _ = Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return (Bool -> FCode Bool) -> Bool -> FCode Bool
forall a b. (a -> b) -> a -> b
$! Bool -> Bool
not (Safety -> Bool
playSafe Safety
safe)
isSimpleOp (StgPrimOp DataToTagOp) _ = Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return Bool
False
isSimpleOp (StgPrimOp op :: PrimOp
op) stg_args :: [StgArg]
stg_args = do
[CmmExpr]
arg_exprs <- [StgArg] -> FCode [CmmExpr]
getNonVoidArgAmodes [StgArg]
stg_args
DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return (Bool -> FCode Bool) -> Bool -> FCode Bool
forall a b. (a -> b) -> a -> b
$! Maybe ([LocalReg] -> FCode ()) -> Bool
forall a. Maybe a -> Bool
isJust (Maybe ([LocalReg] -> FCode ()) -> Bool)
-> Maybe ([LocalReg] -> FCode ()) -> Bool
forall a b. (a -> b) -> a -> b
$ DynFlags -> PrimOp -> [CmmExpr] -> Maybe ([LocalReg] -> FCode ())
shouldInlinePrimOp DynFlags
dflags PrimOp
op [CmmExpr]
arg_exprs
isSimpleOp (StgPrimCallOp _) _ = Bool -> FCode Bool
forall (m :: * -> *) a. Monad m => a -> m a
return Bool
False
chooseReturnBndrs :: Id -> AltType -> [CgStgAlt] -> [NonVoid Id]
chooseReturnBndrs :: Id -> AltType -> [GenStgAlt 'CodeGen] -> [NonVoid Id]
chooseReturnBndrs bndr :: Id
bndr (PrimAlt _) _alts :: [GenStgAlt 'CodeGen]
_alts
= [Id] -> [NonVoid Id]
assertNonVoidIds [Id
bndr]
chooseReturnBndrs _bndr :: Id
_bndr (MultiValAlt n :: ConTagZ
n) [(_, ids :: [BinderP 'CodeGen]
ids, _)]
= ASSERT2(ids `lengthIs` n, ppr n $$ ppr ids $$ ppr _bndr)
[Id] -> [NonVoid Id]
assertNonVoidIds [Id]
[BinderP 'CodeGen]
ids
chooseReturnBndrs bndr :: Id
bndr (AlgAlt _) _alts :: [GenStgAlt 'CodeGen]
_alts
= [Id] -> [NonVoid Id]
assertNonVoidIds [Id
bndr]
chooseReturnBndrs bndr :: Id
bndr PolyAlt _alts :: [GenStgAlt 'CodeGen]
_alts
= [Id] -> [NonVoid Id]
assertNonVoidIds [Id
bndr]
chooseReturnBndrs _ _ _ = String -> [NonVoid Id]
forall a. String -> a
panic "chooseReturnBndrs"
cgAlts :: (GcPlan,ReturnKind) -> NonVoid Id -> AltType -> [CgStgAlt]
-> FCode ReturnKind
cgAlts :: (GcPlan, ReturnKind)
-> NonVoid Id
-> AltType
-> [GenStgAlt 'CodeGen]
-> FCode ReturnKind
cgAlts gc_plan :: (GcPlan, ReturnKind)
gc_plan _bndr :: NonVoid Id
_bndr PolyAlt [(_, _, rhs :: CgStgExpr
rhs)]
= (GcPlan, ReturnKind) -> FCode ReturnKind -> FCode ReturnKind
forall a. (GcPlan, ReturnKind) -> FCode a -> FCode a
maybeAltHeapCheck (GcPlan, ReturnKind)
gc_plan (CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
rhs)
cgAlts gc_plan :: (GcPlan, ReturnKind)
gc_plan _bndr :: NonVoid Id
_bndr (MultiValAlt _) [(_, _, rhs :: CgStgExpr
rhs)]
= (GcPlan, ReturnKind) -> FCode ReturnKind -> FCode ReturnKind
forall a. (GcPlan, ReturnKind) -> FCode a -> FCode a
maybeAltHeapCheck (GcPlan, ReturnKind)
gc_plan (CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
rhs)
cgAlts gc_plan :: (GcPlan, ReturnKind)
gc_plan bndr :: NonVoid Id
bndr (PrimAlt _) alts :: [GenStgAlt 'CodeGen]
alts
= do { DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; [(AltCon, CmmAGraphScoped)]
tagged_cmms <- (GcPlan, ReturnKind)
-> NonVoid Id
-> [GenStgAlt 'CodeGen]
-> FCode [(AltCon, CmmAGraphScoped)]
cgAltRhss (GcPlan, ReturnKind)
gc_plan NonVoid Id
bndr [GenStgAlt 'CodeGen]
alts
; let bndr_reg :: CmmReg
bndr_reg = LocalReg -> CmmReg
CmmLocal (DynFlags -> NonVoid Id -> LocalReg
idToReg DynFlags
dflags NonVoid Id
bndr)
(DEFAULT,deflt :: CmmAGraphScoped
deflt) = [(AltCon, CmmAGraphScoped)] -> (AltCon, CmmAGraphScoped)
forall a. [a] -> a
head [(AltCon, CmmAGraphScoped)]
tagged_cmms
tagged_cmms' :: [(Literal, CmmAGraphScoped)]
tagged_cmms' = [(Literal
lit,CmmAGraphScoped
code)
| (LitAlt lit :: Literal
lit, code :: CmmAGraphScoped
code) <- [(AltCon, CmmAGraphScoped)]
tagged_cmms]
; CmmExpr
-> [(Literal, CmmAGraphScoped)] -> CmmAGraphScoped -> FCode ()
emitCmmLitSwitch (CmmReg -> CmmExpr
CmmReg CmmReg
bndr_reg) [(Literal, CmmAGraphScoped)]
tagged_cmms' CmmAGraphScoped
deflt
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
AssignedDirectly }
cgAlts gc_plan :: (GcPlan, ReturnKind)
gc_plan bndr :: NonVoid Id
bndr (AlgAlt tycon :: TyCon
tycon) alts :: [GenStgAlt 'CodeGen]
alts
= do { DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; (mb_deflt :: Maybe CmmAGraphScoped
mb_deflt, branches :: [(ConTagZ, CmmAGraphScoped)]
branches) <- (GcPlan, ReturnKind)
-> NonVoid Id
-> [GenStgAlt 'CodeGen]
-> FCode (Maybe CmmAGraphScoped, [(ConTagZ, CmmAGraphScoped)])
cgAlgAltRhss (GcPlan, ReturnKind)
gc_plan NonVoid Id
bndr [GenStgAlt 'CodeGen]
alts
; let !fam_sz :: ConTagZ
fam_sz = TyCon -> ConTagZ
tyConFamilySize TyCon
tycon
!bndr_reg :: CmmReg
bndr_reg = LocalReg -> CmmReg
CmmLocal (DynFlags -> NonVoid Id -> LocalReg
idToReg DynFlags
dflags NonVoid Id
bndr)
!ptag_expr :: CmmExpr
ptag_expr = DynFlags -> CmmExpr -> CmmExpr
cmmConstrTag1 DynFlags
dflags (CmmReg -> CmmExpr
CmmReg CmmReg
bndr_reg)
!branches' :: [(ConTagZ, CmmAGraphScoped)]
branches' = (ConTagZ -> ConTagZ)
-> (ConTagZ, CmmAGraphScoped) -> (ConTagZ, CmmAGraphScoped)
forall (a :: * -> * -> *) b c d.
Arrow a =>
a b c -> a (b, d) (c, d)
first ConTagZ -> ConTagZ
forall a. Enum a => a -> a
succ ((ConTagZ, CmmAGraphScoped) -> (ConTagZ, CmmAGraphScoped))
-> [(ConTagZ, CmmAGraphScoped)] -> [(ConTagZ, CmmAGraphScoped)]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [(ConTagZ, CmmAGraphScoped)]
branches
!maxpt :: ConTagZ
maxpt = DynFlags -> ConTagZ
mAX_PTR_TAG DynFlags
dflags
(![(ConTagZ, CmmAGraphScoped)]
via_ptr, ![(ConTagZ, CmmAGraphScoped)]
via_info) = ((ConTagZ, CmmAGraphScoped) -> Bool)
-> [(ConTagZ, CmmAGraphScoped)]
-> ([(ConTagZ, CmmAGraphScoped)], [(ConTagZ, CmmAGraphScoped)])
forall a. (a -> Bool) -> [a] -> ([a], [a])
partition ((ConTagZ -> ConTagZ -> Bool
forall a. Ord a => a -> a -> Bool
< ConTagZ
maxpt) (ConTagZ -> Bool)
-> ((ConTagZ, CmmAGraphScoped) -> ConTagZ)
-> (ConTagZ, CmmAGraphScoped)
-> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (ConTagZ, CmmAGraphScoped) -> ConTagZ
forall a b. (a, b) -> a
fst) [(ConTagZ, CmmAGraphScoped)]
branches'
!small :: Bool
small = DynFlags -> ConTagZ -> Bool
isSmallFamily DynFlags
dflags ConTagZ
fam_sz
; if Bool
small Bool -> Bool -> Bool
|| [(ConTagZ, CmmAGraphScoped)] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [(ConTagZ, CmmAGraphScoped)]
via_info
then
CmmExpr
-> [(ConTagZ, CmmAGraphScoped)]
-> Maybe CmmAGraphScoped
-> ConTagZ
-> ConTagZ
-> FCode ()
emitSwitch CmmExpr
ptag_expr [(ConTagZ, CmmAGraphScoped)]
branches' Maybe CmmAGraphScoped
mb_deflt 1
(if Bool
small then ConTagZ
fam_sz else ConTagZ
maxpt)
else
do
let !untagged_ptr :: CmmExpr
untagged_ptr = DynFlags -> CmmExpr -> CmmExpr
cmmUntag DynFlags
dflags (CmmReg -> CmmExpr
CmmReg CmmReg
bndr_reg)
!itag_expr :: CmmExpr
itag_expr = DynFlags -> CmmExpr -> CmmExpr
getConstrTag DynFlags
dflags CmmExpr
untagged_ptr
!info0 :: [(ConTagZ, CmmAGraphScoped)]
info0 = (ConTagZ -> ConTagZ)
-> (ConTagZ, CmmAGraphScoped) -> (ConTagZ, CmmAGraphScoped)
forall (a :: * -> * -> *) b c d.
Arrow a =>
a b c -> a (b, d) (c, d)
first ConTagZ -> ConTagZ
forall a. Enum a => a -> a
pred ((ConTagZ, CmmAGraphScoped) -> (ConTagZ, CmmAGraphScoped))
-> [(ConTagZ, CmmAGraphScoped)] -> [(ConTagZ, CmmAGraphScoped)]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [(ConTagZ, CmmAGraphScoped)]
via_info
if [(ConTagZ, CmmAGraphScoped)] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [(ConTagZ, CmmAGraphScoped)]
via_ptr then
CmmExpr
-> [(ConTagZ, CmmAGraphScoped)]
-> Maybe CmmAGraphScoped
-> ConTagZ
-> ConTagZ
-> FCode ()
emitSwitch CmmExpr
itag_expr [(ConTagZ, CmmAGraphScoped)]
info0 Maybe CmmAGraphScoped
mb_deflt 0 (ConTagZ
fam_sz ConTagZ -> ConTagZ -> ConTagZ
forall a. Num a => a -> a -> a
- 1)
else do
BlockId
infos_lbl <- FCode BlockId
forall (m :: * -> *). MonadUnique m => m BlockId
newBlockId
CmmTickScope
infos_scp <- FCode CmmTickScope
getTickScope
let spillover :: (ConTagZ, CmmAGraphScoped)
spillover = (ConTagZ
maxpt, (BlockId -> CmmAGraph
mkBranch BlockId
infos_lbl, CmmTickScope
infos_scp))
(mb_shared_deflt :: Maybe CmmAGraphScoped
mb_shared_deflt, mb_shared_branch :: Maybe CmmAGraphScoped
mb_shared_branch) <- case Maybe CmmAGraphScoped
mb_deflt of
(Just (stmts :: CmmAGraph
stmts, scp :: CmmTickScope
scp)) ->
do BlockId
lbl <- FCode BlockId
forall (m :: * -> *). MonadUnique m => m BlockId
newBlockId
(Maybe CmmAGraphScoped, Maybe CmmAGraphScoped)
-> FCode (Maybe CmmAGraphScoped, Maybe CmmAGraphScoped)
forall (m :: * -> *) a. Monad m => a -> m a
return ( CmmAGraphScoped -> Maybe CmmAGraphScoped
forall a. a -> Maybe a
Just (BlockId -> CmmTickScope -> CmmAGraph
mkLabel BlockId
lbl CmmTickScope
scp CmmAGraph -> CmmAGraph -> CmmAGraph
<*> CmmAGraph
stmts, CmmTickScope
scp)
, CmmAGraphScoped -> Maybe CmmAGraphScoped
forall a. a -> Maybe a
Just (BlockId -> CmmAGraph
mkBranch BlockId
lbl, CmmTickScope
scp))
_ -> (Maybe CmmAGraphScoped, Maybe CmmAGraphScoped)
-> FCode (Maybe CmmAGraphScoped, Maybe CmmAGraphScoped)
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe CmmAGraphScoped
forall a. Maybe a
Nothing, Maybe CmmAGraphScoped
forall a. Maybe a
Nothing)
CmmExpr
-> [(ConTagZ, CmmAGraphScoped)]
-> Maybe CmmAGraphScoped
-> ConTagZ
-> ConTagZ
-> FCode ()
emitSwitch CmmExpr
ptag_expr ((ConTagZ, CmmAGraphScoped)
spillover (ConTagZ, CmmAGraphScoped)
-> [(ConTagZ, CmmAGraphScoped)] -> [(ConTagZ, CmmAGraphScoped)]
forall a. a -> [a] -> [a]
: [(ConTagZ, CmmAGraphScoped)]
via_ptr) Maybe CmmAGraphScoped
mb_shared_deflt 1 ConTagZ
maxpt
BlockId
join_lbl <- FCode BlockId
forall (m :: * -> *). MonadUnique m => m BlockId
newBlockId
CmmAGraph -> FCode ()
emit (BlockId -> CmmAGraph
mkBranch BlockId
join_lbl)
BlockId -> FCode ()
emitLabel BlockId
infos_lbl
CmmExpr
-> [(ConTagZ, CmmAGraphScoped)]
-> Maybe CmmAGraphScoped
-> ConTagZ
-> ConTagZ
-> FCode ()
emitSwitch CmmExpr
itag_expr [(ConTagZ, CmmAGraphScoped)]
info0 Maybe CmmAGraphScoped
mb_shared_branch
(ConTagZ
maxpt ConTagZ -> ConTagZ -> ConTagZ
forall a. Num a => a -> a -> a
- 1) (ConTagZ
fam_sz ConTagZ -> ConTagZ -> ConTagZ
forall a. Num a => a -> a -> a
- 1)
BlockId -> FCode ()
emitLabel BlockId
join_lbl
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
AssignedDirectly }
cgAlts _ _ _ _ = String -> FCode ReturnKind
forall a. String -> a
panic "cgAlts"
cgAlgAltRhss :: (GcPlan,ReturnKind) -> NonVoid Id -> [CgStgAlt]
-> FCode ( Maybe CmmAGraphScoped
, [(ConTagZ, CmmAGraphScoped)] )
cgAlgAltRhss :: (GcPlan, ReturnKind)
-> NonVoid Id
-> [GenStgAlt 'CodeGen]
-> FCode (Maybe CmmAGraphScoped, [(ConTagZ, CmmAGraphScoped)])
cgAlgAltRhss gc_plan :: (GcPlan, ReturnKind)
gc_plan bndr :: NonVoid Id
bndr alts :: [GenStgAlt 'CodeGen]
alts
= do { [(AltCon, CmmAGraphScoped)]
tagged_cmms <- (GcPlan, ReturnKind)
-> NonVoid Id
-> [GenStgAlt 'CodeGen]
-> FCode [(AltCon, CmmAGraphScoped)]
cgAltRhss (GcPlan, ReturnKind)
gc_plan NonVoid Id
bndr [GenStgAlt 'CodeGen]
alts
; let { mb_deflt :: Maybe CmmAGraphScoped
mb_deflt = case [(AltCon, CmmAGraphScoped)]
tagged_cmms of
((DEFAULT,rhs :: CmmAGraphScoped
rhs) : _) -> CmmAGraphScoped -> Maybe CmmAGraphScoped
forall a. a -> Maybe a
Just CmmAGraphScoped
rhs
_other :: [(AltCon, CmmAGraphScoped)]
_other -> Maybe CmmAGraphScoped
forall a. Maybe a
Nothing
; branches :: [(ConTagZ, CmmAGraphScoped)]
branches = [ (DataCon -> ConTagZ
dataConTagZ DataCon
con, CmmAGraphScoped
cmm)
| (DataAlt con :: DataCon
con, cmm :: CmmAGraphScoped
cmm) <- [(AltCon, CmmAGraphScoped)]
tagged_cmms ]
}
; (Maybe CmmAGraphScoped, [(ConTagZ, CmmAGraphScoped)])
-> FCode (Maybe CmmAGraphScoped, [(ConTagZ, CmmAGraphScoped)])
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe CmmAGraphScoped
mb_deflt, [(ConTagZ, CmmAGraphScoped)]
branches)
}
cgAltRhss :: (GcPlan,ReturnKind) -> NonVoid Id -> [CgStgAlt]
-> FCode [(AltCon, CmmAGraphScoped)]
cgAltRhss :: (GcPlan, ReturnKind)
-> NonVoid Id
-> [GenStgAlt 'CodeGen]
-> FCode [(AltCon, CmmAGraphScoped)]
cgAltRhss gc_plan :: (GcPlan, ReturnKind)
gc_plan bndr :: NonVoid Id
bndr alts :: [GenStgAlt 'CodeGen]
alts = do
DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
let
base_reg :: LocalReg
base_reg = DynFlags -> NonVoid Id -> LocalReg
idToReg DynFlags
dflags NonVoid Id
bndr
cg_alt :: CgStgAlt -> FCode (AltCon, CmmAGraphScoped)
cg_alt :: GenStgAlt 'CodeGen -> FCode (AltCon, CmmAGraphScoped)
cg_alt (con :: AltCon
con, bndrs :: [BinderP 'CodeGen]
bndrs, rhs :: CgStgExpr
rhs)
= FCode AltCon -> FCode (AltCon, CmmAGraphScoped)
forall a. FCode a -> FCode (a, CmmAGraphScoped)
getCodeScoped (FCode AltCon -> FCode (AltCon, CmmAGraphScoped))
-> FCode AltCon -> FCode (AltCon, CmmAGraphScoped)
forall a b. (a -> b) -> a -> b
$
(GcPlan, ReturnKind) -> FCode AltCon -> FCode AltCon
forall a. (GcPlan, ReturnKind) -> FCode a -> FCode a
maybeAltHeapCheck (GcPlan, ReturnKind)
gc_plan (FCode AltCon -> FCode AltCon) -> FCode AltCon -> FCode AltCon
forall a b. (a -> b) -> a -> b
$
do { [LocalReg]
_ <- AltCon -> LocalReg -> [NonVoid Id] -> FCode [LocalReg]
bindConArgs AltCon
con LocalReg
base_reg ([Id] -> [NonVoid Id]
assertNonVoidIds [Id]
[BinderP 'CodeGen]
bndrs)
; ReturnKind
_ <- CgStgExpr -> FCode ReturnKind
cgExpr CgStgExpr
rhs
; AltCon -> FCode AltCon
forall (m :: * -> *) a. Monad m => a -> m a
return AltCon
con }
[FCode (AltCon, CmmAGraphScoped)]
-> FCode [(AltCon, CmmAGraphScoped)]
forall a. [FCode a] -> FCode [a]
forkAlts (((AltCon, [Id], CgStgExpr) -> FCode (AltCon, CmmAGraphScoped))
-> [(AltCon, [Id], CgStgExpr)] -> [FCode (AltCon, CmmAGraphScoped)]
forall a b. (a -> b) -> [a] -> [b]
map (AltCon, [Id], CgStgExpr) -> FCode (AltCon, CmmAGraphScoped)
GenStgAlt 'CodeGen -> FCode (AltCon, CmmAGraphScoped)
cg_alt [(AltCon, [Id], CgStgExpr)]
[GenStgAlt 'CodeGen]
alts)
maybeAltHeapCheck :: (GcPlan,ReturnKind) -> FCode a -> FCode a
maybeAltHeapCheck :: (GcPlan, ReturnKind) -> FCode a -> FCode a
maybeAltHeapCheck (NoGcInAlts,_) code :: FCode a
code = FCode a
code
maybeAltHeapCheck (GcInAlts regs :: [LocalReg]
regs, AssignedDirectly) code :: FCode a
code =
[LocalReg] -> FCode a -> FCode a
forall a. [LocalReg] -> FCode a -> FCode a
altHeapCheck [LocalReg]
regs FCode a
code
maybeAltHeapCheck (GcInAlts regs :: [LocalReg]
regs, ReturnedTo lret :: BlockId
lret off :: ConTagZ
off) code :: FCode a
code =
[LocalReg] -> BlockId -> ConTagZ -> FCode a -> FCode a
forall a. [LocalReg] -> BlockId -> ConTagZ -> FCode a -> FCode a
altHeapCheckReturnsTo [LocalReg]
regs BlockId
lret ConTagZ
off FCode a
code
cgConApp :: DataCon -> [StgArg] -> FCode ReturnKind
cgConApp :: DataCon -> [StgArg] -> FCode ReturnKind
cgConApp con :: DataCon
con stg_args :: [StgArg]
stg_args
| DataCon -> Bool
isUnboxedTupleCon DataCon
con
= do { [CmmExpr]
arg_exprs <- [StgArg] -> FCode [CmmExpr]
getNonVoidArgAmodes [StgArg]
stg_args
; ConTagZ -> FCode ()
tickyUnboxedTupleReturn ([CmmExpr] -> ConTagZ
forall (t :: * -> *) a. Foldable t => t a -> ConTagZ
length [CmmExpr]
arg_exprs)
; [CmmExpr] -> FCode ReturnKind
emitReturn [CmmExpr]
arg_exprs }
| Bool
otherwise
= ASSERT2( stg_args `lengthIs` countConRepArgs con, ppr con <> parens (ppr (countConRepArgs con)) <+> ppr stg_args )
do { (idinfo :: CgIdInfo
idinfo, fcode_init :: FCode CmmAGraph
fcode_init) <- Id
-> Bool
-> CostCentreStack
-> DataCon
-> [NonVoid StgArg]
-> FCode (CgIdInfo, FCode CmmAGraph)
buildDynCon (DataCon -> Id
dataConWorkId DataCon
con) Bool
False
CostCentreStack
currentCCS DataCon
con ([StgArg] -> [NonVoid StgArg]
assertNonVoidStgArgs [StgArg]
stg_args)
; CmmAGraph -> FCode ()
emit (CmmAGraph -> FCode ()) -> FCode CmmAGraph -> FCode ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< FCode CmmAGraph
fcode_init
; ConTagZ -> FCode ()
tickyReturnNewCon ([StgArg] -> ConTagZ
forall (t :: * -> *) a. Foldable t => t a -> ConTagZ
length [StgArg]
stg_args)
; [CmmExpr] -> FCode ReturnKind
emitReturn [CgIdInfo -> CmmExpr
idInfoToAmode CgIdInfo
idinfo] }
cgIdApp :: Id -> [StgArg] -> FCode ReturnKind
cgIdApp :: Id -> [StgArg] -> FCode ReturnKind
cgIdApp fun_id :: Id
fun_id args :: [StgArg]
args = do
DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
CgIdInfo
fun_info <- Id -> FCode CgIdInfo
getCgIdInfo Id
fun_id
Maybe SelfLoopInfo
self_loop_info <- FCode (Maybe SelfLoopInfo)
getSelfLoop
let cg_fun_id :: Id
cg_fun_id = CgIdInfo -> Id
cg_id CgIdInfo
fun_info
fun_arg :: StgArg
fun_arg = Id -> StgArg
StgVarArg Id
cg_fun_id
fun_name :: Name
fun_name = Id -> Name
idName Id
cg_fun_id
fun :: CmmExpr
fun = CgIdInfo -> CmmExpr
idInfoToAmode CgIdInfo
fun_info
lf_info :: LambdaFormInfo
lf_info = CgIdInfo -> LambdaFormInfo
cg_lf CgIdInfo
fun_info
n_args :: ConTagZ
n_args = [StgArg] -> ConTagZ
forall (t :: * -> *) a. Foldable t => t a -> ConTagZ
length [StgArg]
args
v_args :: ConTagZ
v_args = [StgArg] -> ConTagZ
forall (t :: * -> *) a. Foldable t => t a -> ConTagZ
length ([StgArg] -> ConTagZ) -> [StgArg] -> ConTagZ
forall a b. (a -> b) -> a -> b
$ (StgArg -> Bool) -> [StgArg] -> [StgArg]
forall a. (a -> Bool) -> [a] -> [a]
filter (Type -> Bool
isVoidTy (Type -> Bool) -> (StgArg -> Type) -> StgArg -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. StgArg -> Type
stgArgType) [StgArg]
args
node_points :: DynFlags -> Bool
node_points dflags :: DynFlags
dflags = DynFlags -> LambdaFormInfo -> Bool
nodeMustPointToIt DynFlags
dflags LambdaFormInfo
lf_info
case DynFlags
-> Name
-> Id
-> LambdaFormInfo
-> ConTagZ
-> ConTagZ
-> CgLoc
-> Maybe SelfLoopInfo
-> CallMethod
getCallMethod DynFlags
dflags Name
fun_name Id
cg_fun_id LambdaFormInfo
lf_info ConTagZ
n_args ConTagZ
v_args (CgIdInfo -> CgLoc
cg_loc CgIdInfo
fun_info) Maybe SelfLoopInfo
self_loop_info of
ReturnIt
| Type -> Bool
isVoidTy (Id -> Type
idType Id
fun_id) -> [CmmExpr] -> FCode ReturnKind
emitReturn []
| Bool
otherwise -> [CmmExpr] -> FCode ReturnKind
emitReturn [CmmExpr
fun]
EnterIt -> ASSERT( null args )
CmmExpr -> FCode ReturnKind
emitEnter CmmExpr
fun
SlowCall -> do
{ LambdaFormInfo -> [StgArg] -> FCode ()
tickySlowCall LambdaFormInfo
lf_info [StgArg]
args
; FastString -> FCode ()
emitComment (FastString -> FCode ()) -> FastString -> FCode ()
forall a b. (a -> b) -> a -> b
$ String -> FastString
mkFastString "slowCall"
; CmmExpr -> [StgArg] -> FCode ReturnKind
slowCall CmmExpr
fun [StgArg]
args }
DirectEntry lbl :: CLabel
lbl arity :: ConTagZ
arity -> do
{ ConTagZ -> [StgArg] -> FCode ()
tickyDirectCall ConTagZ
arity [StgArg]
args
; if DynFlags -> Bool
node_points DynFlags
dflags
then Convention -> CLabel -> ConTagZ -> [StgArg] -> FCode ReturnKind
directCall Convention
NativeNodeCall CLabel
lbl ConTagZ
arity (StgArg
fun_argStgArg -> [StgArg] -> [StgArg]
forall a. a -> [a] -> [a]
:[StgArg]
args)
else Convention -> CLabel -> ConTagZ -> [StgArg] -> FCode ReturnKind
directCall Convention
NativeDirectCall CLabel
lbl ConTagZ
arity [StgArg]
args }
JumpToIt blk_id :: BlockId
blk_id lne_regs :: [LocalReg]
lne_regs -> do
{ FCode ()
adjustHpBackwards
; [CmmExpr]
cmm_args <- [StgArg] -> FCode [CmmExpr]
getNonVoidArgAmodes [StgArg]
args
; [LocalReg] -> [CmmExpr] -> FCode ()
emitMultiAssign [LocalReg]
lne_regs [CmmExpr]
cmm_args
; CmmAGraph -> FCode ()
emit (BlockId -> CmmAGraph
mkBranch BlockId
blk_id)
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
AssignedDirectly }
emitEnter :: CmmExpr -> FCode ReturnKind
emitEnter :: CmmExpr -> FCode ReturnKind
emitEnter fun :: CmmExpr
fun = do
{ DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; FCode ()
adjustHpBackwards
; Sequel
sequel <- FCode Sequel
getSequel
; ConTagZ
updfr_off <- FCode ConTagZ
getUpdFrameOff
; case Sequel
sequel of
Return -> do
{ let entry :: CmmExpr
entry = DynFlags -> CmmExpr -> CmmExpr
entryCode DynFlags
dflags (CmmExpr -> CmmExpr) -> CmmExpr -> CmmExpr
forall a b. (a -> b) -> a -> b
$ DynFlags -> CmmExpr -> CmmExpr
closureInfoPtr DynFlags
dflags (CmmExpr -> CmmExpr) -> CmmExpr -> CmmExpr
forall a b. (a -> b) -> a -> b
$ CmmReg -> CmmExpr
CmmReg CmmReg
nodeReg
; CmmAGraph -> FCode ()
emit (CmmAGraph -> FCode ()) -> CmmAGraph -> FCode ()
forall a b. (a -> b) -> a -> b
$ DynFlags
-> Convention -> CmmExpr -> [CmmExpr] -> ConTagZ -> CmmAGraph
mkJump DynFlags
dflags Convention
NativeNodeCall CmmExpr
entry
[DynFlags -> CmmExpr -> CmmExpr
cmmUntag DynFlags
dflags CmmExpr
fun] ConTagZ
updfr_off
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return ReturnKind
AssignedDirectly
}
AssignTo res_regs :: [LocalReg]
res_regs _ -> do
{ BlockId
lret <- FCode BlockId
forall (m :: * -> *). MonadUnique m => m BlockId
newBlockId
; let (off :: ConTagZ
off, _, copyin :: CmmAGraph
copyin) = DynFlags
-> Convention
-> Area
-> [LocalReg]
-> [LocalReg]
-> (ConTagZ, [GlobalReg], CmmAGraph)
copyInOflow DynFlags
dflags Convention
NativeReturn (BlockId -> Area
Young BlockId
lret) [LocalReg]
res_regs []
; BlockId
lcall <- FCode BlockId
forall (m :: * -> *). MonadUnique m => m BlockId
newBlockId
; ConTagZ
updfr_off <- FCode ConTagZ
getUpdFrameOff
; let area :: Area
area = BlockId -> Area
Young BlockId
lret
; let (outArgs :: ConTagZ
outArgs, regs :: [GlobalReg]
regs, copyout :: CmmAGraph
copyout) = DynFlags
-> Convention
-> Transfer
-> Area
-> [CmmExpr]
-> ConTagZ
-> [CmmExpr]
-> (ConTagZ, [GlobalReg], CmmAGraph)
copyOutOflow DynFlags
dflags Convention
NativeNodeCall Transfer
Call Area
area
[CmmExpr
fun] ConTagZ
updfr_off []
; let entry :: CmmExpr
entry = DynFlags -> CmmExpr -> CmmExpr
entryCode DynFlags
dflags (DynFlags -> CmmExpr -> CmmExpr
closureInfoPtr DynFlags
dflags (CmmReg -> CmmExpr
CmmReg CmmReg
nodeReg))
the_call :: CmmAGraph
the_call = CmmExpr
-> Maybe BlockId
-> ConTagZ
-> ConTagZ
-> ConTagZ
-> [GlobalReg]
-> CmmAGraph
toCall CmmExpr
entry (BlockId -> Maybe BlockId
forall a. a -> Maybe a
Just BlockId
lret) ConTagZ
updfr_off ConTagZ
off ConTagZ
outArgs [GlobalReg]
regs
; CmmTickScope
tscope <- FCode CmmTickScope
getTickScope
; CmmAGraph -> FCode ()
emit (CmmAGraph -> FCode ()) -> CmmAGraph -> FCode ()
forall a b. (a -> b) -> a -> b
$
CmmAGraph
copyout CmmAGraph -> CmmAGraph -> CmmAGraph
<*>
CmmExpr -> BlockId -> BlockId -> Maybe Bool -> CmmAGraph
mkCbranch (DynFlags -> CmmExpr -> CmmExpr
cmmIsTagged DynFlags
dflags (CmmReg -> CmmExpr
CmmReg CmmReg
nodeReg))
BlockId
lret BlockId
lcall Maybe Bool
forall a. Maybe a
Nothing CmmAGraph -> CmmAGraph -> CmmAGraph
<*>
BlockId -> CmmAGraphScoped -> CmmAGraph
outOfLine BlockId
lcall (CmmAGraph
the_call,CmmTickScope
tscope) CmmAGraph -> CmmAGraph -> CmmAGraph
<*>
BlockId -> CmmTickScope -> CmmAGraph
mkLabel BlockId
lret CmmTickScope
tscope CmmAGraph -> CmmAGraph -> CmmAGraph
<*>
CmmAGraph
copyin
; ReturnKind -> FCode ReturnKind
forall (m :: * -> *) a. Monad m => a -> m a
return (BlockId -> ConTagZ -> ReturnKind
ReturnedTo BlockId
lret ConTagZ
off)
}
}
cgTick :: Tickish Id -> FCode ()
cgTick :: Tickish Id -> FCode ()
cgTick tick :: Tickish Id
tick
= do { DynFlags
dflags <- FCode DynFlags
forall (m :: * -> *). HasDynFlags m => m DynFlags
getDynFlags
; case Tickish Id
tick of
ProfNote cc :: CostCentre
cc t :: Bool
t p :: Bool
p -> CostCentre -> Bool -> Bool -> FCode ()
emitSetCCC CostCentre
cc Bool
t Bool
p
HpcTick m :: Module
m n :: ConTagZ
n -> CmmAGraph -> FCode ()
emit (DynFlags -> Module -> ConTagZ -> CmmAGraph
mkTickBox DynFlags
dflags Module
m ConTagZ
n)
SourceNote s :: RealSrcSpan
s n :: String
n -> CmmTickish -> FCode ()
emitTick (CmmTickish -> FCode ()) -> CmmTickish -> FCode ()
forall a b. (a -> b) -> a -> b
$ RealSrcSpan -> String -> CmmTickish
forall id. RealSrcSpan -> String -> Tickish id
SourceNote RealSrcSpan
s String
n
_other :: Tickish Id
_other -> () -> FCode ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
}