{-# OPTIONS_GHC -fno-warn-orphans #-}
module Futhark.Analysis.PrimExp.Convert
(
primExpToExp
, primExpFromExp
, primExpFromSubExp
, primExpFromSubExpM
, replaceInPrimExp
, replaceInPrimExpM
, substituteInPrimExp
, module Futhark.Analysis.PrimExp
) where
import qualified Control.Monad.Fail as Fail
import Control.Monad.Identity
import Data.Loc
import qualified Data.Map.Strict as M
import Data.Maybe
import Futhark.Analysis.PrimExp
import Futhark.Construct
import Futhark.Representation.AST
primExpToExp :: MonadBinder m =>
(v -> m (Exp (Lore m))) -> PrimExp v -> m (Exp (Lore m))
primExpToExp f (BinOpExp op x y) =
BasicOp <$> (BinOp op
<$> primExpToSubExp "binop_x" f x
<*> primExpToSubExp "binop_y" f y)
primExpToExp f (CmpOpExp op x y) =
BasicOp <$> (CmpOp op
<$> primExpToSubExp "cmpop_x" f x
<*> primExpToSubExp "cmpop_y" f y)
primExpToExp f (UnOpExp op x) =
BasicOp <$> (UnOp op <$> primExpToSubExp "unop_x" f x)
primExpToExp f (ConvOpExp op x) =
BasicOp <$> (ConvOp op <$> primExpToSubExp "convop_x" f x)
primExpToExp _ (ValueExp v) =
return $ BasicOp $ SubExp $ Constant v
primExpToExp f (FunExp h args t) =
Apply (nameFromString h) <$> args' <*> pure [primRetType t] <*> pure (Safe, noLoc, [])
where args' = zip <$> mapM (primExpToSubExp "apply_arg" f) args <*> pure (repeat Observe)
primExpToExp f (LeafExp v _) =
f v
instance ToExp v => ToExp (PrimExp v) where
toExp = primExpToExp toExp
primExpToSubExp :: MonadBinder m =>
String -> (v -> m (Exp (Lore m))) -> PrimExp v -> m SubExp
primExpToSubExp s f e = letSubExp s =<< primExpToExp f e
primExpFromExp :: (Fail.MonadFail m, Annotations lore) =>
(VName -> m (PrimExp v)) -> Exp lore -> m (PrimExp v)
primExpFromExp f (BasicOp (BinOp op x y)) =
BinOpExp op <$> primExpFromSubExpM f x <*> primExpFromSubExpM f y
primExpFromExp f (BasicOp (CmpOp op x y)) =
CmpOpExp op <$> primExpFromSubExpM f x <*> primExpFromSubExpM f y
primExpFromExp f (BasicOp (UnOp op x)) =
UnOpExp op <$> primExpFromSubExpM f x
primExpFromExp f (BasicOp (ConvOp op x)) =
ConvOpExp op <$> primExpFromSubExpM f x
primExpFromExp _ (BasicOp (SubExp (Constant v))) =
return $ ValueExp v
primExpFromExp f (Apply fname args ts _)
| isBuiltInFunction fname, [Prim t] <- retTypeValues ts =
FunExp (nameToString fname) <$> mapM (primExpFromSubExpM f . fst) args <*> pure t
primExpFromExp _ _ = fail "Not a PrimExp"
primExpFromSubExpM :: Fail.MonadFail m =>
(VName -> m (PrimExp v)) -> SubExp -> m (PrimExp v)
primExpFromSubExpM f (Var v) = f v
primExpFromSubExpM _ (Constant v) = return $ ValueExp v
primExpFromSubExp :: PrimType -> SubExp -> PrimExp VName
primExpFromSubExp t (Var v) = LeafExp v t
primExpFromSubExp _ (Constant v) = ValueExp v
replaceInPrimExpM :: Monad m =>
(a -> PrimType -> m (PrimExp b)) ->
PrimExp a -> m (PrimExp b)
replaceInPrimExpM f (LeafExp v pt) =
f v pt
replaceInPrimExpM _ (ValueExp v) =
return $ ValueExp v
replaceInPrimExpM f (BinOpExp bop pe1 pe2) =
constFoldPrimExp <$>
(BinOpExp bop <$> replaceInPrimExpM f pe1 <*> replaceInPrimExpM f pe2)
replaceInPrimExpM f (CmpOpExp cop pe1 pe2) =
CmpOpExp cop <$> replaceInPrimExpM f pe1 <*> replaceInPrimExpM f pe2
replaceInPrimExpM f (UnOpExp uop pe) =
UnOpExp uop <$> replaceInPrimExpM f pe
replaceInPrimExpM f (ConvOpExp cop pe) =
ConvOpExp cop <$> replaceInPrimExpM f pe
replaceInPrimExpM f (FunExp h args t) =
FunExp h <$> mapM (replaceInPrimExpM f) args <*> pure t
replaceInPrimExp :: (a -> PrimType -> PrimExp b) ->
PrimExp a -> PrimExp b
replaceInPrimExp f e = runIdentity $ replaceInPrimExpM f' e
where f' x y = return $ f x y
substituteInPrimExp :: Ord v => M.Map v (PrimExp v)
-> PrimExp v -> PrimExp v
substituteInPrimExp tab = replaceInPrimExp $ \v t ->
fromMaybe (LeafExp v t) $ M.lookup v tab