{-# LANGUAGE CPP #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE PatternGuards #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TupleSections #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE ViewPatterns #-}
module Data.Array.Accelerate.Trafo.Algebra (
evalPrimApp
) where
import Prelude hiding ( exp )
import Data.Bits
import Data.Char
import Data.Monoid
import GHC.Float ( float2Double, double2Float )
import Text.PrettyPrint
import Unsafe.Coerce
import qualified Prelude as P
import Data.Array.Accelerate.AST
import Data.Array.Accelerate.Type
import Data.Array.Accelerate.Product
import Data.Array.Accelerate.Pretty.Print ( prettyPrim )
import Data.Array.Accelerate.Array.Sugar ( (:.)(..), Elt(..), Tuple(..), fromTuple )
import Data.Array.Accelerate.Analysis.Match
import Data.Array.Accelerate.Trafo.Base
import qualified Data.Array.Accelerate.Debug as Stats
propagate
:: forall acc env aenv exp. Kit acc
=> Gamma acc env env aenv
-> PreOpenExp acc env aenv exp
-> Maybe exp
propagate env = cvtE
where
cvtE :: PreOpenExp acc env aenv e -> Maybe e
cvtE exp = case exp of
Const c -> Just (toElt c)
PrimConst c -> Just (evalPrimConst c)
Prj ix (Var v) | Tuple t <- prjExp v env -> cvtT ix t
Prj ix e | Just c <- cvtE e -> cvtP ix (fromTuple c)
Var ix
| e <- prjExp ix env
, Nothing <- match exp e -> cvtE e
IndexHead (cvtE -> Just (_ :. z)) -> Just z
IndexTail (cvtE -> Just (sh :. _)) -> Just sh
_ -> Nothing
cvtP :: TupleIdx t e -> t -> Maybe e
cvtP ZeroTupIdx (_, v) = Just v
cvtP (SuccTupIdx idx) (tup, _) = cvtP idx tup
cvtT :: TupleIdx t e -> Tuple (PreOpenExp acc env aenv) t -> Maybe e
cvtT ZeroTupIdx (SnocTup _ e) = cvtE e
cvtT (SuccTupIdx idx) (SnocTup tup _) = cvtT idx tup
#if __GLASGOW_HASKELL__ < 800
cvtT _ _ = error "hey what's the head angle on that thing?"
#endif
evalPrimApp
:: forall acc env aenv a r. (Kit acc, Elt a, Elt r)
=> Gamma acc env env aenv
-> PrimFun (a -> r)
-> PreOpenExp acc env aenv a
-> (Any, PreOpenExp acc env aenv r)
evalPrimApp env f x
| Just r <- commutes f x env = evalPrimApp env f r
| otherwise
= maybe (Any False, PrimApp f x) (Any True,)
$ case f of
PrimAdd ty -> evalAdd ty x env
PrimSub ty -> evalSub ty x env
PrimMul ty -> evalMul ty x env
PrimNeg ty -> evalNeg ty x env
PrimAbs ty -> evalAbs ty x env
PrimSig ty -> evalSig ty x env
PrimQuot ty -> evalQuot ty x env
PrimRem ty -> evalRem ty x env
PrimQuotRem ty -> evalQuotRem ty x env
PrimIDiv ty -> evalIDiv ty x env
PrimMod ty -> evalMod ty x env
PrimDivMod ty -> evalDivMod ty x env
PrimBAnd ty -> evalBAnd ty x env
PrimBOr ty -> evalBOr ty x env
PrimBXor ty -> evalBXor ty x env
PrimBNot ty -> evalBNot ty x env
PrimBShiftL ty -> evalBShiftL ty x env
PrimBShiftR ty -> evalBShiftR ty x env
PrimBRotateL ty -> evalBRotateL ty x env
PrimBRotateR ty -> evalBRotateR ty x env
PrimPopCount ty -> evalPopCount ty x env
PrimCountLeadingZeros ty -> evalCountLeadingZeros ty x env
PrimCountTrailingZeros ty -> evalCountTrailingZeros ty x env
PrimFDiv ty -> evalFDiv ty x env
PrimRecip ty -> evalRecip ty x env
PrimSin ty -> evalSin ty x env
PrimCos ty -> evalCos ty x env
PrimTan ty -> evalTan ty x env
PrimAsin ty -> evalAsin ty x env
PrimAcos ty -> evalAcos ty x env
PrimAtan ty -> evalAtan ty x env
PrimSinh ty -> evalSinh ty x env
PrimCosh ty -> evalCosh ty x env
PrimTanh ty -> evalTanh ty x env
PrimAsinh ty -> evalAsinh ty x env
PrimAcosh ty -> evalAcosh ty x env
PrimAtanh ty -> evalAtanh ty x env
PrimExpFloating ty -> evalExpFloating ty x env
PrimSqrt ty -> evalSqrt ty x env
PrimLog ty -> evalLog ty x env
PrimFPow ty -> evalFPow ty x env
PrimLogBase ty -> evalLogBase ty x env
PrimAtan2 ty -> evalAtan2 ty x env
PrimTruncate ta tb -> evalTruncate ta tb x env
PrimRound ta tb -> evalRound ta tb x env
PrimFloor ta tb -> evalFloor ta tb x env
PrimCeiling ta tb -> evalCeiling ta tb x env
PrimIsNaN ty -> evalIsNaN ty x env
PrimIsInfinite ty -> evalIsInfinite ty x env
PrimLt ty -> evalLt ty x env
PrimGt ty -> evalGt ty x env
PrimLtEq ty -> evalLtEq ty x env
PrimGtEq ty -> evalGtEq ty x env
PrimEq ty -> evalEq ty x env
PrimNEq ty -> evalNEq ty x env
PrimMax ty -> evalMax ty x env
PrimMin ty -> evalMin ty x env
PrimLAnd -> evalLAnd x env
PrimLOr -> evalLOr x env
PrimLNot -> evalLNot x env
PrimOrd -> evalOrd x env
PrimChr -> evalChr x env
PrimBoolToInt -> evalBoolToInt x env
PrimFromIntegral ta tb -> evalFromIntegral ta tb x env
PrimToFloating ta tb -> evalToFloating ta tb x env
PrimCoerce ta tb -> evalCoerce ta tb x env
commutes
:: forall acc env aenv a r. Kit acc
=> PrimFun (a -> r)
-> PreOpenExp acc env aenv a
-> Gamma acc env env aenv
-> Maybe (PreOpenExp acc env aenv a)
commutes f x env = case f of
PrimAdd _ -> swizzle x
PrimMul _ -> swizzle x
PrimBAnd _ -> swizzle x
PrimBOr _ -> swizzle x
PrimBXor _ -> swizzle x
PrimEq _ -> swizzle x
PrimNEq _ -> swizzle x
PrimMax _ -> swizzle x
PrimMin _ -> swizzle x
_ -> Nothing
where
swizzle :: PreOpenExp acc env aenv (b,b) -> Maybe (PreOpenExp acc env aenv (b,b))
swizzle (Tuple (NilTup `SnocTup` a `SnocTup` b))
| Nothing <- propagate env a
, Just _ <- propagate env b
= Stats.ruleFired (pprFun "commutes" f)
$ Just $ Tuple (NilTup `SnocTup` b `SnocTup` a)
swizzle _
= Nothing
type a :-> b = forall acc env aenv. Kit acc => PreOpenExp acc env aenv a -> Gamma acc env env aenv -> Maybe (PreOpenExp acc env aenv b)
eval1 :: Elt b => (a -> b) -> a :-> b
eval1 f x env
| Just a <- propagate env x = Stats.substitution "constant fold" . Just $ Const (fromElt (f a))
| otherwise = Nothing
eval2 :: Elt c => (a -> b -> c) -> (a,b) :-> c
eval2 f (untup2 -> Just (x,y)) env
| Just a <- propagate env x
, Just b <- propagate env y
= Stats.substitution "constant fold"
$ Just $ Const (fromElt (f a b))
eval2 _ _ _
= Nothing
tup2 :: (Elt a, Elt b) => (PreOpenExp acc env aenv a, PreOpenExp acc env aenv b) -> PreOpenExp acc env aenv (a, b)
tup2 (a,b) = Tuple (NilTup `SnocTup` a `SnocTup` b)
untup2 :: PreOpenExp acc env aenv (a, b) -> Maybe (PreOpenExp acc env aenv a, PreOpenExp acc env aenv b)
untup2 exp
| Tuple (NilTup `SnocTup` a `SnocTup` b) <- exp = Just (a, b)
| otherwise = Nothing
pprFun :: String -> PrimFun f -> String
pprFun rule f = show $ text rule <+> snd (prettyPrim f)
evalAdd :: Elt a => NumType a -> (a,a) :-> a
evalAdd (IntegralNumType ty) | IntegralDict <- integralDict ty = evalAdd'
evalAdd (FloatingNumType ty) | FloatingDict <- floatingDict ty = evalAdd'
evalAdd' :: (Elt a, Eq a, Num a) => (a,a) :-> a
evalAdd' (untup2 -> Just (x,y)) env
| Just a <- propagate env x
, a == 0
= Stats.ruleFired "x+0" $ Just y
evalAdd' arg env
= eval2 (+) arg env
evalSub :: Elt a => NumType a -> (a,a) :-> a
evalSub ty@(IntegralNumType ty') | IntegralDict <- integralDict ty' = evalSub' ty
evalSub ty@(FloatingNumType ty') | FloatingDict <- floatingDict ty' = evalSub' ty
evalSub' :: forall a. (Elt a, Eq a, Num a) => NumType a -> (a,a) :-> a
evalSub' ty (untup2 -> Just (x,y)) env
| Just b <- propagate env y
, b == 0
= Stats.ruleFired "x-0" $ Just x
| Nothing <- propagate env x
, Just b <- propagate env y
= Stats.ruleFired "-y+x"
$ Just . snd $ evalPrimApp env (PrimAdd ty) (Tuple $ NilTup `SnocTup` Const (fromElt (-b)) `SnocTup` x)
| Just Refl <- match x y
= Stats.ruleFired "x-x"
$ Just $ Const (fromElt (0::a))
evalSub' _ arg env
= eval2 (-) arg env
evalMul :: Elt a => NumType a -> (a,a) :-> a
evalMul (IntegralNumType ty) | IntegralDict <- integralDict ty = evalMul'
evalMul (FloatingNumType ty) | FloatingDict <- floatingDict ty = evalMul'
evalMul' :: (Elt a, Eq a, Num a) => (a,a) :-> a
evalMul' (untup2 -> Just (x,y)) env
| Just a <- propagate env x
, Nothing <- propagate env y
= case a of
0 -> Stats.ruleFired "x*0" $ Just x
1 -> Stats.ruleFired "x*1" $ Just y
_ -> Nothing
evalMul' arg env
= eval2 (*) arg env
evalNeg :: Elt a => NumType a -> a :-> a
evalNeg _ x _ | PrimApp PrimNeg{} x' <- x = Stats.ruleFired "negate/negate" $ Just x'
evalNeg (IntegralNumType ty) x env | IntegralDict <- integralDict ty = eval1 negate x env
evalNeg (FloatingNumType ty) x env | FloatingDict <- floatingDict ty = eval1 negate x env
evalAbs :: Elt a => NumType a -> a :-> a
evalAbs (IntegralNumType ty) | IntegralDict <- integralDict ty = eval1 abs
evalAbs (FloatingNumType ty) | FloatingDict <- floatingDict ty = eval1 abs
evalSig :: Elt a => NumType a -> a :-> a
evalSig (IntegralNumType ty) | IntegralDict <- integralDict ty = eval1 signum
evalSig (FloatingNumType ty) | FloatingDict <- floatingDict ty = eval1 signum
evalQuot :: IntegralType a -> (a,a) :-> a
evalQuot ty exp env
| Just qr <- evalQuotRem ty exp env
, Just (q,_) <- untup2 qr
= Just q
evalQuot _ _ _
= Nothing
evalRem :: IntegralType a -> (a,a) :-> a
evalRem ty exp env
| Just qr <- evalQuotRem ty exp env
, Just (_,r) <- untup2 qr
= Just r
evalRem _ _ _
= Nothing
evalQuotRem :: forall a. IntegralType a -> (a,a) :-> (a,a)
evalQuotRem ty exp env
| IntegralDict <- integralDict ty
, Tuple (NilTup `SnocTup` x `SnocTup` y) <- exp
, Just b <- propagate env y
= case b of
0 -> Nothing
1 -> Stats.ruleFired "quotRem x 1" $ Just (tup2 (x, Const (fromElt (0::a))))
_ -> case propagate env x of
Nothing -> Nothing
Just a -> Stats.substitution "constant fold"
$ Just $ let (u,v) = quotRem a b
in tup2 (Const (fromElt u), Const (fromElt v))
evalQuotRem _ _ _
= Nothing
evalIDiv :: IntegralType a -> (a,a) :-> a
evalIDiv ty exp env
| Just dm <- evalDivMod ty exp env
, Just (d,_) <- untup2 dm
= Just d
evalIDiv _ _ _
= Nothing
evalMod :: IntegralType a -> (a,a) :-> a
evalMod ty exp env
| Just dm <- evalDivMod ty exp env
, Just (_,m) <- untup2 dm
= Just m
evalMod _ _ _
= Nothing
evalDivMod :: forall a. IntegralType a -> (a,a) :-> (a,a)
evalDivMod ty exp env
| IntegralDict <- integralDict ty
, Tuple (NilTup `SnocTup` x `SnocTup` y) <- exp
, Just b <- propagate env y
= case b of
0 -> Nothing
1 -> Stats.ruleFired "divMod x 1" $ Just (tup2 (x, Const (fromElt (0::a))))
_ -> case propagate env x of
Nothing -> Nothing
Just a -> Stats.substitution "constant fold"
$ Just $ let (u,v) = divMod a b
in tup2 (Const (fromElt u), Const (fromElt v))
evalDivMod _ _ _
= Nothing
evalBAnd :: Elt a => IntegralType a -> (a,a) :-> a
evalBAnd ty | IntegralDict <- integralDict ty = eval2 (.&.)
evalBOr :: Elt a => IntegralType a -> (a,a) :-> a
evalBOr ty | IntegralDict <- integralDict ty = eval2 (.|.)
evalBXor :: Elt a => IntegralType a -> (a,a) :-> a
evalBXor ty | IntegralDict <- integralDict ty = eval2 xor
evalBNot :: Elt a => IntegralType a -> a :-> a
evalBNot ty | IntegralDict <- integralDict ty = eval1 complement
evalBShiftL :: Elt a => IntegralType a -> (a,Int) :-> a
evalBShiftL _ (untup2 -> Just (x,i)) env
| Just 0 <- propagate env i
= Stats.ruleFired "x `shiftL` 0" $ Just x
evalBShiftL ty arg env
| IntegralDict <- integralDict ty = eval2 shiftL arg env
evalBShiftR :: Elt a => IntegralType a -> (a,Int) :-> a
evalBShiftR _ (untup2 -> Just (x,i)) env
| Just 0 <- propagate env i
= Stats.ruleFired "x `shiftR` 0" $ Just x
evalBShiftR ty arg env
| IntegralDict <- integralDict ty = eval2 shiftR arg env
evalBRotateL :: Elt a => IntegralType a -> (a,Int) :-> a
evalBRotateL _ (untup2 -> Just (x,i)) env
| Just 0 <- propagate env i
= Stats.ruleFired "x `rotateL` 0" $ Just x
evalBRotateL ty arg env
| IntegralDict <- integralDict ty = eval2 rotateL arg env
evalBRotateR :: Elt a => IntegralType a -> (a,Int) :-> a
evalBRotateR _ (untup2 -> Just (x,i)) env
| Just 0 <- propagate env i
= Stats.ruleFired "x `rotateR` 0" $ Just x
evalBRotateR ty arg env
| IntegralDict <- integralDict ty = eval2 rotateR arg env
evalPopCount :: IntegralType a -> a :-> Int
evalPopCount ty | IntegralDict <- integralDict ty = eval1 popCount
evalCountLeadingZeros :: IntegralType a -> a :-> Int
#if __GLASGOW_HASKELL__ >= 710
evalCountLeadingZeros ty | IntegralDict <- integralDict ty = eval1 countLeadingZeros
#else
evalCountLeadingZeros ty | IntegralDict <- integralDict ty = eval1 clz
where
clz x = (w-1) - go (w-1)
where
go i | i < 0 = i
| testBit x i = i
| otherwise = go (i-1)
w = finiteBitSize x
#endif
evalCountTrailingZeros :: IntegralType a -> a :-> Int
#if __GLASGOW_HASKELL__ >= 710
evalCountTrailingZeros ty | IntegralDict <- integralDict ty = eval1 countTrailingZeros
#else
evalCountTrailingZeros ty | IntegralDict <- integralDict ty = eval1 ctz
where
ctz x = go 0
where
go i | i >= w = i
| testBit x i = i
| otherwise = go (i+1)
w = finiteBitSize x
#endif
evalFDiv :: Elt a => FloatingType a -> (a,a) :-> a
evalFDiv ty | FloatingDict <- floatingDict ty = evalFDiv'
evalFDiv' :: (Elt a, Fractional a, Eq a) => (a,a) :-> a
evalFDiv' (untup2 -> Just (x,y)) env
| Just 1 <- propagate env y
= Stats.ruleFired "x/1" $ Just x
evalFDiv' arg env
= eval2 (/) arg env
evalRecip :: Elt a => FloatingType a -> a :-> a
evalRecip ty | FloatingDict <- floatingDict ty = eval1 recip
evalSin :: Elt a => FloatingType a -> a :-> a
evalSin ty | FloatingDict <- floatingDict ty = eval1 sin
evalCos :: Elt a => FloatingType a -> a :-> a
evalCos ty | FloatingDict <- floatingDict ty = eval1 cos
evalTan :: Elt a => FloatingType a -> a :-> a
evalTan ty | FloatingDict <- floatingDict ty = eval1 tan
evalAsin :: Elt a => FloatingType a -> a :-> a
evalAsin ty | FloatingDict <- floatingDict ty = eval1 asin
evalAcos :: Elt a => FloatingType a -> a :-> a
evalAcos ty | FloatingDict <- floatingDict ty = eval1 acos
evalAtan :: Elt a => FloatingType a -> a :-> a
evalAtan ty | FloatingDict <- floatingDict ty = eval1 atan
evalSinh :: Elt a => FloatingType a -> a :-> a
evalSinh ty | FloatingDict <- floatingDict ty = eval1 sinh
evalCosh :: Elt a => FloatingType a -> a :-> a
evalCosh ty | FloatingDict <- floatingDict ty = eval1 cosh
evalTanh :: Elt a => FloatingType a -> a :-> a
evalTanh ty | FloatingDict <- floatingDict ty = eval1 tanh
evalAsinh :: Elt a => FloatingType a -> a :-> a
evalAsinh ty | FloatingDict <- floatingDict ty = eval1 asinh
evalAcosh :: Elt a => FloatingType a -> a :-> a
evalAcosh ty | FloatingDict <- floatingDict ty = eval1 acosh
evalAtanh :: Elt a => FloatingType a -> a :-> a
evalAtanh ty | FloatingDict <- floatingDict ty = eval1 atanh
evalExpFloating :: Elt a => FloatingType a -> a :-> a
evalExpFloating ty | FloatingDict <- floatingDict ty = eval1 P.exp
evalSqrt :: Elt a => FloatingType a -> a :-> a
evalSqrt ty | FloatingDict <- floatingDict ty = eval1 sqrt
evalLog :: Elt a => FloatingType a -> a :-> a
evalLog ty | FloatingDict <- floatingDict ty = eval1 log
evalFPow :: Elt a => FloatingType a -> (a,a) :-> a
evalFPow ty | FloatingDict <- floatingDict ty = eval2 (**)
evalLogBase :: Elt a => FloatingType a -> (a,a) :-> a
evalLogBase ty | FloatingDict <- floatingDict ty = eval2 logBase
evalAtan2 :: Elt a => FloatingType a -> (a,a) :-> a
evalAtan2 ty | FloatingDict <- floatingDict ty = eval2 atan2
evalTruncate :: Elt b => FloatingType a -> IntegralType b -> a :-> b
evalTruncate ta tb
| FloatingDict <- floatingDict ta
, IntegralDict <- integralDict tb = eval1 truncate
evalRound :: Elt b => FloatingType a -> IntegralType b -> a :-> b
evalRound ta tb
| FloatingDict <- floatingDict ta
, IntegralDict <- integralDict tb = eval1 round
evalFloor :: Elt b => FloatingType a -> IntegralType b -> a :-> b
evalFloor ta tb
| FloatingDict <- floatingDict ta
, IntegralDict <- integralDict tb = eval1 floor
evalCeiling :: Elt b => FloatingType a -> IntegralType b -> a :-> b
evalCeiling ta tb
| FloatingDict <- floatingDict ta
, IntegralDict <- integralDict tb = eval1 ceiling
evalIsNaN :: FloatingType a -> a :-> Bool
evalIsNaN ty | FloatingDict <- floatingDict ty = eval1 isNaN
evalIsInfinite :: FloatingType a -> a :-> Bool
evalIsInfinite ty | FloatingDict <- floatingDict ty = eval1 isInfinite
evalLt :: ScalarType a -> (a,a) :-> Bool
evalLt (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 (<)
evalLt (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 (<)
evalLt (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 (<)
evalGt :: ScalarType a -> (a,a) :-> Bool
evalGt (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 (>)
evalGt (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 (>)
evalGt (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 (>)
evalLtEq :: ScalarType a -> (a,a) :-> Bool
evalLtEq (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 (<=)
evalLtEq (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 (<=)
evalLtEq (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 (<=)
evalGtEq :: ScalarType a -> (a,a) :-> Bool
evalGtEq (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 (>=)
evalGtEq (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 (>=)
evalGtEq (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 (>=)
evalEq :: ScalarType a -> (a,a) :-> Bool
evalEq (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 (==)
evalEq (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 (==)
evalEq (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 (==)
evalNEq :: ScalarType a -> (a,a) :-> Bool
evalNEq (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 (/=)
evalNEq (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 (/=)
evalNEq (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 (/=)
evalMax :: Elt a => ScalarType a -> (a,a) :-> a
evalMax (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 max
evalMax (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 max
evalMax (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 max
evalMin :: Elt a => ScalarType a -> (a,a) :-> a
evalMin (NumScalarType (IntegralNumType ty)) | IntegralDict <- integralDict ty = eval2 min
evalMin (NumScalarType (FloatingNumType ty)) | FloatingDict <- floatingDict ty = eval2 min
evalMin (NonNumScalarType ty) | NonNumDict <- nonNumDict ty = eval2 min
evalLAnd :: (Bool,Bool) :-> Bool
evalLAnd (untup2 -> Just (x,y)) env
| Just a <- propagate env x
= Just $ if a then Stats.ruleFired "True &&" y
else Stats.ruleFired "False &&" $ Const (fromElt False)
evalLAnd _ _
= Nothing
evalLOr :: (Bool,Bool) :-> Bool
evalLOr (untup2 -> Just (x,y)) env
| Just a <- propagate env x
= Just $ if a then Stats.ruleFired "True ||" $ Const (fromElt True)
else Stats.ruleFired "False ||" y
evalLOr _ _
= Nothing
evalLNot :: Bool :-> Bool
evalLNot x _ | PrimApp PrimLNot x' <- x = Stats.ruleFired "not/not" $ Just x'
evalLNot x env = eval1 not x env
evalOrd :: Char :-> Int
evalOrd = eval1 ord
evalChr :: Int :-> Char
evalChr = eval1 chr
evalBoolToInt :: Bool :-> Int
evalBoolToInt = eval1 fromEnum
evalFromIntegral :: Elt b => IntegralType a -> NumType b -> a :-> b
evalFromIntegral ta (IntegralNumType tb)
| IntegralDict <- integralDict ta
, IntegralDict <- integralDict tb = eval1 fromIntegral
evalFromIntegral ta (FloatingNumType tb)
| IntegralDict <- integralDict ta
, FloatingDict <- floatingDict tb = eval1 fromIntegral
evalToFloating :: Elt b => NumType a -> FloatingType b -> a :-> b
evalToFloating (IntegralNumType ta) tb x env
| IntegralDict <- integralDict ta
, FloatingDict <- floatingDict tb = eval1 realToFrac x env
evalToFloating (FloatingNumType ta) tb x env
| TypeFloat FloatingDict <- ta
, TypeFloat FloatingDict <- tb = Just x
| TypeDouble FloatingDict <- ta
, TypeDouble FloatingDict <- tb = Just x
| TypeFloat FloatingDict <- ta
, TypeDouble FloatingDict <- tb = eval1 float2Double x env
| TypeDouble FloatingDict <- ta
, TypeFloat FloatingDict <- tb = eval1 double2Float x env
| FloatingDict <- floatingDict ta
, FloatingDict <- floatingDict tb = eval1 realToFrac x env
evalCoerce :: Elt b => ScalarType a -> ScalarType b -> a :-> b
evalCoerce _ _ = eval1 unsafeCoerce
evalPrimConst :: PrimConst a -> a
evalPrimConst (PrimMinBound ty) = evalMinBound ty
evalPrimConst (PrimMaxBound ty) = evalMaxBound ty
evalPrimConst (PrimPi ty) = evalPi ty
evalMinBound :: BoundedType a -> a
evalMinBound (IntegralBoundedType ty) | IntegralDict <- integralDict ty = minBound
evalMinBound (NonNumBoundedType ty) | NonNumDict <- nonNumDict ty = minBound
evalMaxBound :: BoundedType a -> a
evalMaxBound (IntegralBoundedType ty) | IntegralDict <- integralDict ty = maxBound
evalMaxBound (NonNumBoundedType ty) | NonNumDict <- nonNumDict ty = maxBound
evalPi :: FloatingType a -> a
evalPi ty | FloatingDict <- floatingDict ty = pi