module Language.SMTLib2.Internals.Expression where
import Language.SMTLib2.Internals.Type hiding (Field)
import Language.SMTLib2.Internals.Type.Nat
import Language.SMTLib2.Internals.Type.List (List(..))
import qualified Language.SMTLib2.Internals.Type.List as List
import Data.Typeable
import Text.Show
import Data.GADT.Compare
import Data.GADT.Show
import Data.Functor.Identity
import Data.Ratio
type family AllEq (tp :: Type) (n :: Nat) :: [Type] where
AllEq tp Z = '[]
AllEq tp (S n) = tp ': (AllEq tp n)
allEqToList :: Natural n -> List a (AllEq tp n) -> [a tp]
allEqToList Zero Nil = []
allEqToList (Succ n) (x ::: xs) = x:allEqToList n xs
allEqFromList :: [a tp] -> (forall n. Natural n -> List a (AllEq tp n) -> r) -> r
allEqFromList [] f = f Zero Nil
allEqFromList (x:xs) f = allEqFromList xs (\n arg -> f (Succ n) (x ::: arg))
allEqOf :: Repr tp -> Natural n -> List Repr (AllEq tp n)
allEqOf tp Zero = Nil
allEqOf tp (Succ n) = tp ::: allEqOf tp n
mapAllEq :: Monad m => (e1 tp -> m (e2 tp))
-> Natural n
-> List e1 (AllEq tp n)
-> m (List e2 (AllEq tp n))
mapAllEq f Zero Nil = return Nil
mapAllEq f (Succ n) (x ::: xs) = do
x' <- f x
xs' <- mapAllEq f n xs
return (x' ::: xs')
data Function (fun :: ([Type],Type) -> *) (con :: ([Type],*) -> *) (field :: (*,Type) -> *) (sig :: ([Type],Type)) where
Fun :: fun '(arg,res) -> Function fun con field '(arg,res)
Eq :: Repr tp -> Natural n -> Function fun con field '(AllEq tp n,BoolType)
Distinct :: Repr tp -> Natural n -> Function fun con field '(AllEq tp n,BoolType)
Map :: List Repr idx -> Function fun con field '(arg,res)
-> Function fun con field '(Lifted arg idx,ArrayType idx res)
Ord :: NumRepr tp -> OrdOp -> Function fun con field '([tp,tp],BoolType)
Arith :: NumRepr tp -> ArithOp -> Natural n
-> Function fun con field '(AllEq tp n,tp)
ArithIntBin :: ArithOpInt -> Function fun con field '([IntType,IntType],IntType)
Divide :: Function fun con field '([RealType,RealType],RealType)
Abs :: NumRepr tp -> Function fun con field '( '[tp],tp)
Not :: Function fun con field '( '[BoolType],BoolType)
Logic :: LogicOp -> Natural n -> Function fun con field '(AllEq BoolType n,BoolType)
ToReal :: Function fun con field '( '[IntType],RealType)
ToInt :: Function fun con field '( '[RealType],IntType)
ITE :: Repr a -> Function fun con field '([BoolType,a,a],a)
BVComp :: BVCompOp -> Natural a -> Function fun con field '([BitVecType a,BitVecType a],BoolType)
BVBin :: BVBinOp -> Natural a -> Function fun con field '([BitVecType a,BitVecType a],BitVecType a)
BVUn :: BVUnOp -> Natural a -> Function fun con field '( '[BitVecType a],BitVecType a)
Select :: List Repr idx -> Repr val -> Function fun con field '(ArrayType idx val ': idx,val)
Store :: List Repr idx -> Repr val -> Function fun con field '(ArrayType idx val ': val ': idx,ArrayType idx val)
ConstArray :: List Repr idx -> Repr val -> Function fun con field '( '[val],ArrayType idx val)
Concat :: Natural n1 -> Natural n2 -> Function fun con field '([BitVecType n1,BitVecType n2],BitVecType (n1 + n2))
Extract :: (((start + len) <= bw) ~ True)
=> Natural bw -> Natural start -> Natural len -> Function fun con field '( '[BitVecType bw],BitVecType len)
Constructor :: IsDatatype a => con '(arg,a)
-> Function fun con field '(arg,DataType a)
Test :: (IsDatatype a) => con '(arg,a) -> Function fun con field '( '[DataType a],BoolType)
Field :: (IsDatatype a) => field '(a,t) -> Function fun con field '( '[DataType a],t)
Divisible :: Integer -> Function fun con field '( '[IntType],BoolType)
data AnyFunction (fun :: ([Type],Type) -> *) (con :: ([Type],*) -> *) (field :: (*,Type) -> *) where
AnyFunction :: Function fun con field '(arg,t) -> AnyFunction fun con field
data OrdOp = Ge | Gt | Le | Lt deriving (Eq,Ord,Show)
data ArithOp = Plus | Mult | Minus deriving (Eq,Ord,Show)
data ArithOpInt = Div | Mod | Rem deriving (Eq,Ord,Show)
data LogicOp = And | Or | XOr | Implies deriving (Eq,Ord,Show)
data BVCompOp = BVULE
| BVULT
| BVUGE
| BVUGT
| BVSLE
| BVSLT
| BVSGE
| BVSGT
deriving (Eq,Ord,Show)
data BVBinOp = BVAdd
| BVSub
| BVMul
| BVURem
| BVSRem
| BVUDiv
| BVSDiv
| BVSHL
| BVLSHR
| BVASHR
| BVXor
| BVAnd
| BVOr
deriving (Eq,Ord,Show)
data BVUnOp = BVNot | BVNeg deriving (Eq,Ord,Show)
data LetBinding (v :: Type -> *) (e :: Type -> *) (t :: Type)
= LetBinding { letVar :: v t
, letExpr :: e t }
data Quantifier = Forall | Exists deriving (Typeable,Eq,Ord,Show)
data Expression (v :: Type -> *) (qv :: Type -> *) (fun :: ([Type],Type) -> *) (con :: ([Type],*) -> *) (field :: (*,Type) -> *) (fv :: Type -> *) (lv :: Type -> *) (e :: Type -> *) (res :: Type) where
#if __GLASGOW_HASKELL__>=712
#endif
Var :: v res -> Expression v qv fun con field fv lv e res
#if __GLASGOW_HASKELL__>=712
#endif
QVar :: qv res -> Expression v qv fun con field fv lv e res
#if __GLASGOW_HASKELL__>=712
#endif
FVar :: fv res -> Expression v qv fun con field fv lv e res
#if __GLASGOW_HASKELL__>=712
#endif
LVar :: lv res -> Expression v qv fun con field fv lv e res
#if __GLASGOW_HASKELL__>=712
#endif
App :: Function fun con field '(arg,res)
-> List e arg
-> Expression v qv fun con field fv lv e res
#if __GLASGOW_HASKELL__>=712
#endif
Const :: Value con a -> Expression v qv fun con field fv lv e a
#if __GLASGOW_HASKELL__>=712
#endif
AsArray :: Function fun con field '(arg,res)
-> Expression v qv fun con field fv lv e (ArrayType arg res)
#if __GLASGOW_HASKELL__>=712
#endif
Quantification :: Quantifier -> List qv arg -> e BoolType
-> Expression v qv fun con field fv lv e BoolType
#if __GLASGOW_HASKELL__>=712
#endif
Let :: List (LetBinding lv e) arg
-> e res
-> Expression v qv fun con field fv lv e res
instance (GEq fun,GEq con,GEq field)
=> Eq (Function fun con field sig) where
(==) = defaultEq
class SMTOrd (t :: Type) where
lt :: Function fun con field '( '[t,t],BoolType)
le :: Function fun con field '( '[t,t],BoolType)
gt :: Function fun con field '( '[t,t],BoolType)
ge :: Function fun con field '( '[t,t],BoolType)
instance SMTOrd IntType where
lt = Ord NumInt Lt
le = Ord NumInt Le
gt = Ord NumInt Gt
ge = Ord NumInt Ge
instance SMTOrd RealType where
lt = Ord NumReal Lt
le = Ord NumReal Le
gt = Ord NumReal Gt
ge = Ord NumReal Ge
class SMTArith t where
arithFromInteger :: Integer -> ConcreteValue t
arith :: ArithOp -> Natural n -> Function fun con field '(AllEq t n,t)
plus :: Natural n -> Function fun con field '(AllEq t n,t)
minus :: Natural n -> Function fun con field '(AllEq t n,t)
mult :: Natural n -> Function fun con field '(AllEq t n,t)
abs' :: Function fun con field '( '[t],t)
instance SMTArith IntType where
arithFromInteger n = IntValueC n
arith = Arith NumInt
plus = Arith NumInt Plus
minus = Arith NumInt Minus
mult = Arith NumInt Mult
abs' = Abs NumInt
instance SMTArith RealType where
arithFromInteger n = RealValueC (fromInteger n)
arith = Arith NumReal
plus = Arith NumReal Plus
minus = Arith NumReal Minus
mult = Arith NumReal Mult
abs' = Abs NumReal
functionType :: Monad m
=> (forall arg t. fun '(arg,t) -> m (List Repr arg,Repr t))
-> (forall arg dt. IsDatatype dt => con '(arg,dt)
-> m (List Repr arg,Datatype '(DatatypeSig dt,dt)))
-> (forall dt t. IsDatatype dt => field '(dt,t)
-> m (Datatype '(DatatypeSig dt,dt),Repr t))
-> Function fun con field '(arg,res)
-> m (List Repr arg,Repr res)
functionType f _ _ (Fun fun) = f fun
functionType _ _ _ (Eq tp n) = return (allEqOf tp n,BoolRepr)
functionType _ _ _ (Distinct tp n) = return (allEqOf tp n,BoolRepr)
functionType f g h (Map idx fun) = do
(arg,res) <- functionType f g h fun
return (liftType arg idx,ArrayRepr idx res)
functionType _ _ _ (Ord tp _) = return (numRepr tp ::: numRepr tp ::: Nil,BoolRepr)
functionType _ _ _ (Arith tp _ n) = return (allEqOf (numRepr tp) n,numRepr tp)
functionType _ _ _ (ArithIntBin _) = return (IntRepr ::: IntRepr ::: Nil,IntRepr)
functionType _ _ _ Divide = return (RealRepr ::: RealRepr ::: Nil,RealRepr)
functionType _ _ _ (Abs tp) = return (numRepr tp ::: Nil,numRepr tp)
functionType _ _ _ Not = return (BoolRepr ::: Nil,BoolRepr)
functionType _ _ _ (Logic op n) = return (allEqOf BoolRepr n,BoolRepr)
functionType _ _ _ ToReal = return (IntRepr ::: Nil,RealRepr)
functionType _ _ _ ToInt = return (RealRepr ::: Nil,IntRepr)
functionType _ _ _ (ITE tp) = return (BoolRepr ::: tp ::: tp ::: Nil,tp)
functionType _ _ _ (BVComp _ n) = return (BitVecRepr n ::: BitVecRepr n ::: Nil,BoolRepr)
functionType _ _ _ (BVBin _ n) = return (BitVecRepr n ::: BitVecRepr n ::: Nil,BitVecRepr n)
functionType _ _ _ (BVUn _ n) = return (BitVecRepr n ::: Nil,BitVecRepr n)
functionType _ _ _ (Select idx el) = return (ArrayRepr idx el ::: idx,el)
functionType _ _ _ (Store idx el) = return (ArrayRepr idx el ::: el ::: idx,ArrayRepr idx el)
functionType _ _ _ (ConstArray idx el) = return (el ::: Nil,ArrayRepr idx el)
functionType _ _ _ (Concat bw1 bw2) = return (BitVecRepr bw1 ::: BitVecRepr bw2 ::: Nil,
BitVecRepr (naturalAdd bw1 bw2))
functionType _ _ _ (Extract bw start len) = return (BitVecRepr bw ::: Nil,BitVecRepr len)
functionType _ f _ (Constructor con) = do
(tps,dt) <- f con
return (tps,DataRepr dt)
functionType _ f _ (Test con) = do
(_,dt) <- f con
return (DataRepr dt ::: Nil,BoolRepr)
functionType _ _ f (Field field) = do
(dt,tp) <- f field
return (DataRepr dt ::: Nil,tp)
functionType _ _ _ (Divisible _) = return (IntRepr ::: Nil,BoolRepr)
expressionType :: (Monad m,Functor m)
=> (forall t. v t -> m (Repr t))
-> (forall t. qv t -> m (Repr t))
-> (forall arg t. fun '(arg,t) -> m (List Repr arg,Repr t))
-> (forall arg dt. IsDatatype dt => con '(arg,dt)
-> m (List Repr arg,Datatype '(DatatypeSig dt,dt)))
-> (forall dt t. IsDatatype dt => field '(dt,t)
-> m (Datatype '(DatatypeSig dt,dt),Repr t))
-> (forall t. fv t -> m (Repr t))
-> (forall t. lv t -> m (Repr t))
-> (forall t. e t -> m (Repr t))
-> Expression v qv fun con field fv lv e res
-> m (Repr res)
expressionType f _ _ _ _ _ _ _ (Var v) = f v
expressionType _ f _ _ _ _ _ _ (QVar v) = f v
expressionType _ _ _ _ _ f _ _ (FVar v) = f v
expressionType _ _ _ _ _ _ f _ (LVar v) = f v
expressionType _ _ f g h _ _ _ (App fun arg) = fmap snd $ functionType f g h fun
expressionType _ _ _ _ _ _ _ _ (Const v) = return $ valueType v
expressionType _ _ f g h _ _ _ (AsArray fun) = do
(arg,res) <- functionType f g h fun
return $ ArrayRepr arg res
expressionType _ _ _ _ _ _ _ _ (Quantification _ _ _) = return BoolRepr
expressionType _ _ _ _ _ _ _ f (Let _ body) = f body
mapExpr :: (Functor m,Monad m,Typeable con2)
=> (forall t. v1 t -> m (v2 t))
-> (forall t. qv1 t -> m (qv2 t))
-> (forall arg t. fun1 '(arg,t) -> m (fun2 '(arg,t)))
-> (forall arg t. con1 '(arg,t) -> m (con2 '(arg,t)))
-> (forall t res. field1 '(t,res) -> m (field2 '(t,res)))
-> (forall t. fv1 t -> m (fv2 t))
-> (forall t. lv1 t -> m (lv2 t))
-> (forall t. e1 t -> m (e2 t))
-> Expression v1 qv1 fun1 con1 field1 fv1 lv1 e1 r
-> m (Expression v2 qv2 fun2 con2 field2 fv2 lv2 e2 r)
mapExpr f _ _ _ _ _ _ _ (Var v) = fmap Var (f v)
mapExpr _ f _ _ _ _ _ _ (QVar v) = fmap QVar (f v)
mapExpr _ _ _ _ _ f _ _ (FVar v) = fmap FVar (f v)
mapExpr _ _ _ _ _ _ f _ (LVar v) = fmap LVar (f v)
mapExpr _ _ f g h _ _ i (App fun args) = do
fun' <- mapFunction f g h fun
args' <- List.mapM i args
return (App fun' args')
mapExpr _ _ _ f _ _ _ _ (Const val) = fmap Const (mapValue f val)
mapExpr _ _ f g h _ _ _ (AsArray fun) = fmap AsArray (mapFunction f g h fun)
mapExpr _ f _ _ _ _ _ g (Quantification q args body) = do
args' <- List.mapM f args
body' <- g body
return (Quantification q args' body')
mapExpr _ _ _ _ _ _ f g (Let args body) = do
args' <- List.mapM (\bind -> do
nv <- f (letVar bind)
nexpr <- g (letExpr bind)
return $ LetBinding nv nexpr
) args
body' <- g body
return (Let args' body')
mapFunction :: (Functor m,Monad m)
=> (forall arg t. fun1 '(arg,t) -> m (fun2 '(arg,t)))
-> (forall arg t. con1 '(arg,t) -> m (con2 '(arg,t)))
-> (forall t res. field1 '(t,res) -> m (field2 '(t,res)))
-> Function fun1 con1 field1 '(arg,res)
-> m (Function fun2 con2 field2 '(arg,res))
mapFunction f _ _ (Fun x) = fmap Fun (f x)
mapFunction _ _ _ (Eq tp n) = return (Eq tp n)
mapFunction _ _ _ (Distinct tp n) = return (Distinct tp n)
mapFunction f g h (Map idx x) = do
x' <- mapFunction f g h x
return (Map idx x')
mapFunction _ _ _ (Ord tp op) = return (Ord tp op)
mapFunction _ _ _ (Arith tp op n) = return (Arith tp op n)
mapFunction _ _ _ (ArithIntBin op) = return (ArithIntBin op)
mapFunction _ _ _ Divide = return Divide
mapFunction _ _ _ (Abs tp) = return (Abs tp)
mapFunction _ _ _ Not = return Not
mapFunction _ _ _ (Logic op n) = return (Logic op n)
mapFunction _ _ _ ToReal = return ToReal
mapFunction _ _ _ ToInt = return ToInt
mapFunction _ _ _ (ITE tp) = return (ITE tp)
mapFunction _ _ _ (BVComp op bw) = return (BVComp op bw)
mapFunction _ _ _ (BVBin op bw) = return (BVBin op bw)
mapFunction _ _ _ (BVUn op bw) = return (BVUn op bw)
mapFunction _ _ _ (Select idx el) = return (Select idx el)
mapFunction _ _ _ (Store idx el) = return (Store idx el)
mapFunction _ _ _ (ConstArray idx el) = return (ConstArray idx el)
mapFunction _ _ _ (Concat bw1 bw2) = return (Concat bw1 bw2)
mapFunction _ _ _ (Extract bw start len) = return (Extract bw start len)
mapFunction _ f _ (Constructor con) = fmap Constructor (f con)
mapFunction _ f _ (Test con) = fmap Test (f con)
mapFunction _ _ f (Field x) = fmap Field (f x)
mapFunction _ _ _ (Divisible x) = return (Divisible x)
instance (GShow v,GShow qv,GShow fun,GShow con,GShow field,GShow fv,GShow lv,GShow e)
=> Show (Expression v qv fun con field fv lv e r) where
showsPrec p (Var v) = showParen (p>10) $
showString "Var " .
gshowsPrec 11 v
showsPrec p (QVar v) = showParen (p>10) $
showString "QVar " .
gshowsPrec 11 v
showsPrec p (FVar v) = showParen (p>10) $
showString "FVar " .
gshowsPrec 11 v
showsPrec p (LVar v) = showParen (p>10) $
showString "LVar " .
gshowsPrec 11 v
showsPrec p (App fun args)
= showParen (p>10) $
showString "App " .
showsPrec 11 fun .
showChar ' ' .
showsPrec 11 args
showsPrec p (Const val) = showsPrec p val
showsPrec p (AsArray fun)
= showParen (p>10) $
showString "AsArray " .
showsPrec 11 fun
showsPrec p (Quantification q args body)
= showParen (p>10) $
showsPrec 11 q .
showChar ' ' .
showsPrec 11 args .
showChar ' ' .
gshowsPrec 11 body
showsPrec p (Let args body)
= showParen (p>10) $
showString "Let " .
showListWith id (runIdentity $ List.toList
(\(LetBinding v e)
-> return $ (gshowsPrec 10 v) . showChar '=' . (gshowsPrec 10 e)
) args) .
showChar ' ' .
gshowsPrec 10 body
instance (GShow v,GShow qv,GShow fun,GShow con,GShow field,GShow fv,GShow lv,GShow e)
=> GShow (Expression v qv fun con field fv lv e) where
gshowsPrec = showsPrec
instance (GShow fun,GShow con,GShow field)
=> Show (Function fun con field sig) where
showsPrec p (Fun x) = gshowsPrec p x
showsPrec _ (Eq _ _) = showString "Eq"
showsPrec _ (Distinct _ _) = showString "Distinct"
showsPrec p (Map _ x) = showParen (p>10) $
showString "Map " .
showsPrec 11 x
showsPrec p (Ord tp op) = showParen (p>10) $
showString "Ord " .
showsPrec 11 tp .
showChar ' ' .
showsPrec 11 op
showsPrec p (Arith tp op _) = showParen (p>10) $
showString "Arith " .
showsPrec 11 tp .
showChar ' ' .
showsPrec 11 op
showsPrec p (ArithIntBin op) = showParen (p>10) $
showString "ArithIntBin " .
showsPrec 11 op
showsPrec p Divide = showString "Divide"
showsPrec p (Abs tp) = showParen (p>10) $
showString "Abs " .
showsPrec 11 tp
showsPrec _ Not = showString "Not"
showsPrec p (Logic op _) = showParen (p>10) $
showString "Logic " .
showsPrec 11 op
showsPrec _ ToReal = showString "ToReal"
showsPrec _ ToInt = showString "ToInt"
showsPrec _ (ITE _) = showString "ITE"
showsPrec p (BVComp op _) = showParen (p>10) $
showString "BVComp " .
showsPrec 11 op
showsPrec p (BVBin op _) = showParen (p>10) $
showString "BVBin " .
showsPrec 11 op
showsPrec p (BVUn op _) = showParen (p>10) $
showString "BVUn " .
showsPrec 11 op
showsPrec _ (Select _ _) = showString "Select"
showsPrec _ (Store _ _) = showString "Store"
showsPrec _ (ConstArray _ _) = showString "ConstArray"
showsPrec _ (Concat _ _) = showString "Concat"
showsPrec p (Extract bw start len)
= showParen (p>10) $
showString "Extract " .
showsPrec 11 bw .
showChar ' ' .
showsPrec 11 start .
showChar ' ' .
showsPrec 11 len
showsPrec p (Constructor con) = showParen (p>10) $
showString "Constructor " .
gshowsPrec 11 con
showsPrec p (Test con) = showParen (p>10) $
showString "Test " .
gshowsPrec 11 con
showsPrec p (Field x) = showParen (p>10) $
showString "Field " .
gshowsPrec 11 x
showsPrec p (Divisible x) = showParen (p>10) $
showString "Divisible " .
showsPrec 11 x
data RenderMode = SMTRendering deriving (Eq,Ord,Show)
renderExprDefault :: (GetType qv,GShow v,GShow qv,GShow fun,GShow con,
GShow field,GShow fv,GShow lv,GShow e)
=> RenderMode
-> Expression v qv fun con field fv lv e tp
-> ShowS
renderExprDefault m
= renderExpr m (gshowsPrec 11) (gshowsPrec 11) (gshowsPrec 11) (gshowsPrec 11)
(gshowsPrec 11) (gshowsPrec 11) (gshowsPrec 11) (gshowsPrec 11)
renderExpr :: (GetType qv) => RenderMode
-> (forall tp. v tp -> ShowS)
-> (forall tp. qv tp -> ShowS)
-> (forall arg res. fun '(arg,res) -> ShowS)
-> (forall arg dt. con '(arg,dt) -> ShowS)
-> (forall dt res. field '(dt,res) -> ShowS)
-> (forall tp. fv tp -> ShowS)
-> (forall tp. lv tp -> ShowS)
-> (forall tp. e tp -> ShowS)
-> Expression v qv fun con field fv lv e tp
-> ShowS
renderExpr _ f _ _ _ _ _ _ _ (Var x) = f x
renderExpr _ _ f _ _ _ _ _ _ (QVar x) = f x
renderExpr _ _ _ _ _ _ f _ _ (FVar x) = f x
renderExpr _ _ _ _ _ _ _ f _ (LVar x) = f x
renderExpr SMTRendering _ _ f g h _ _ i (App fun args)
= showChar '(' .
renderFunction SMTRendering f g h fun .
renderArgs i args .
showChar ')'
where
renderArgs :: (forall tp. e tp -> ShowS) -> List e tps -> ShowS
renderArgs f Nil = id
renderArgs f (x ::: xs) = showChar ' ' . f x . renderArgs f xs
renderExpr m _ _ _ f _ _ _ _ (Const val) = renderValue m f val
renderExpr SMTRendering _ _ f g h _ _ _ (AsArray fun)
= showString "(_ as-array " .
renderFunction SMTRendering f g h fun .
showChar ')'
renderExpr SMTRendering _ f _ _ _ _ _ g (Quantification q args body)
= showChar '(' .
showString (case q of
Forall -> "forall"
Exists -> "exists") .
showString " (" . renderArgs f args . showString ") " . g body . showChar ')'
where
renderArgs :: GetType qv => (forall tp. qv tp -> ShowS)
-> List qv tps -> ShowS
renderArgs _ Nil = id
renderArgs f (x ::: xs) = showChar '(' .
f x . showChar ' ' .
renderType SMTRendering (getType x) .
showChar ')' .
(case xs of
Nil -> id
_ -> showChar ' ' . renderArgs f xs)
renderExpr SMTRendering _ _ _ _ _ _ f g (Let args body)
= showString "(let (" . renderArgs f g args . showString ") " . g body . showChar ')'
where
renderArgs :: (forall tp. lv tp -> ShowS) -> (forall tp. e tp -> ShowS)
-> List (LetBinding lv e) args
-> ShowS
renderArgs _ _ Nil = id
renderArgs f g (x ::: xs)
= showChar '(' .
f (letVar x) . showChar ' ' .
g (letExpr x) . showChar ')' .
(case xs of
Nil -> id
_ -> showChar ' ' . renderArgs f g xs)
renderValue :: RenderMode -> (forall arg dt. con '(arg,dt) -> ShowS) -> Value con tp -> ShowS
renderValue SMTRendering _ (BoolValue v) = if v then showString "true" else showString "false"
renderValue SMTRendering _ (IntValue v)
= if v>=0 then showsPrec 0 v
else showString "(- " .
showsPrec 0 (negate v) .
showChar ')'
renderValue SMTRendering _ (RealValue v)
= showString "(/ " . n . showChar ' ' . d . showChar ')'
where
n = if numerator v >= 0
then showsPrec 0 (numerator v)
else showString "(- " . showsPrec 0 (negate $ numerator v) . showChar ')'
d = showsPrec 0 (denominator v)
renderValue SMTRendering _ (BitVecValue n bw)
= showString "(_ bv" .
showsPrec 0 n .
showChar ' ' .
showsPrec 0 (naturalToInteger bw) .
showChar ')'
renderValue SMTRendering f (ConstrValue con Nil)
= f con
renderValue SMTRendering f (ConstrValue con xs)
= showChar '(' . f con . renderValues f xs . showChar ')'
where
renderValues :: (forall arg dt. con '(arg,dt) -> ShowS) -> List (Value con) arg -> ShowS
renderValues f Nil = id
renderValues f (x ::: xs) = showChar ' ' . renderValue SMTRendering f x . renderValues f xs
renderFunction :: RenderMode
-> (forall arg res. fun '(arg,res) -> ShowS)
-> (forall arg dt. con '(arg,dt) -> ShowS)
-> (forall dt res. field '(dt,res) -> ShowS)
-> Function fun con field '(arg,res)
-> ShowS
renderFunction _ f _ _ (Fun x) = f x
renderFunction SMTRendering _ _ _ (Eq _ _) = showChar '='
renderFunction SMTRendering _ _ _ (Distinct _ _) = showString "distinct"
renderFunction SMTRendering f g h (Map _ fun)
= showString "(map " .
renderFunction SMTRendering f g h fun .
showChar ')'
renderFunction SMTRendering _ _ _ (Ord _ Ge) = showString ">="
renderFunction SMTRendering _ _ _ (Ord _ Gt) = showChar '>'
renderFunction SMTRendering _ _ _ (Ord _ Le) = showString "<="
renderFunction SMTRendering _ _ _ (Ord _ Lt) = showString "<"
renderFunction SMTRendering _ _ _ (Arith _ Plus _) = showChar '+'
renderFunction SMTRendering _ _ _ (Arith _ Mult _) = showChar '*'
renderFunction SMTRendering _ _ _ (Arith _ Minus _) = showChar '-'
renderFunction SMTRendering _ _ _ (ArithIntBin Div) = showString "div"
renderFunction SMTRendering _ _ _ (ArithIntBin Mod) = showString "mod"
renderFunction SMTRendering _ _ _ (ArithIntBin Rem) = showString "rem"
renderFunction SMTRendering _ _ _ Divide = showChar '/'
renderFunction SMTRendering _ _ _ (Abs _) = showString "abs"
renderFunction SMTRendering _ _ _ Not = showString "not"
renderFunction SMTRendering _ _ _ (Logic And _) = showString "and"
renderFunction SMTRendering _ _ _ (Logic Or _) = showString "or"
renderFunction SMTRendering _ _ _ (Logic XOr _) = showString "xor"
renderFunction SMTRendering _ _ _ (Logic Implies _) = showString "=>"
renderFunction SMTRendering _ _ _ ToReal = showString "to_real"
renderFunction SMTRendering _ _ _ ToInt = showString "to_int"
renderFunction SMTRendering _ _ _ (ITE _) = showString "ite"
renderFunction SMTRendering _ _ _ (BVComp op _) = showString $ case op of
BVULE -> "bvule"
BVULT -> "bvult"
BVUGE -> "bvuge"
BVUGT -> "bvugt"
BVSLE -> "bvsle"
BVSLT -> "bvslt"
BVSGE -> "bvsge"
BVSGT -> "bvsgt"
renderFunction SMTRendering _ _ _ (BVBin op _) = showString $ case op of
BVAdd -> "bvadd"
BVSub -> "bvsub"
BVMul -> "bvmul"
BVURem -> "bvurem"
BVSRem -> "bvsrem"
BVUDiv -> "bvudiv"
BVSDiv -> "bvsdiv"
BVSHL -> "bvshl"
BVLSHR -> "bvshr"
BVASHR -> "bvashr"
BVXor -> "bvxor"
BVAnd -> "bvand"
BVOr -> "bvor"
renderFunction SMTRendering _ _ _ (BVUn op _) = showString $ case op of
BVNot -> "bvnot"
BVNeg -> "bvneg"
renderFunction SMTRendering _ _ _ (Select _ _) = showString "select"
renderFunction SMTRendering _ _ _ (Store _ _) = showString "store"
renderFunction SMTRendering _ _ _ (ConstArray idx el)
= showString "(as const " .
renderType SMTRendering (ArrayRepr idx el) .
showChar ')'
renderFunction SMTRendering _ _ _ (Concat _ _) = showString "concat"
renderFunction SMTRendering _ _ _ (Extract _ start len)
= showString "(_ extract " .
showString (show $ start'+len'1) .
showChar ' ' .
showString (show start') .
showChar ')'
where
start' = naturalToInteger start
len' = naturalToInteger len
renderFunction SMTRendering _ f _ (Constructor con) = f con
renderFunction SMTRendering _ f _ (Test con) = showString "is-" . f con
renderFunction SMTRendering _ _ f (Field field) = f field
renderFunction SMTRendering _ _ _ (Divisible n) = showString "(_ divisible " .
showsPrec 10 n .
showChar ')'
renderType :: RenderMode -> Repr tp -> ShowS
renderType SMTRendering BoolRepr = showString "Bool"
renderType SMTRendering IntRepr = showString "Int"
renderType SMTRendering RealRepr = showString "Real"
renderType SMTRendering (BitVecRepr bw) = showString "(BitVec " .
showString (show $ naturalToInteger bw) .
showChar ')'
renderType SMTRendering (ArrayRepr idx el) = showString "(Array (" .
renderTypes idx .
showString ") " .
renderType SMTRendering el .
showChar ')'
where
renderTypes :: List Repr tps -> ShowS
renderTypes Nil = id
renderTypes (tp ::: Nil) = renderType SMTRendering tp
renderTypes (tp ::: tps) = renderType SMTRendering tp .
showChar ' ' .
renderTypes tps
renderType _ (DataRepr dt) = showString (datatypeName dt)
instance (GShow fun,GShow con,GShow field)
=> GShow (Function fun con field) where
gshowsPrec = showsPrec
instance (GEq v,GEq e) => GEq (LetBinding v e) where
geq (LetBinding v1 e1) (LetBinding v2 e2) = do
Refl <- geq v1 v2
geq e1 e2
instance (GCompare v,GCompare e) => GCompare (LetBinding v e) where
gcompare (LetBinding v1 e1) (LetBinding v2 e2) = case gcompare v1 v2 of
GEQ -> gcompare e1 e2
r -> r
instance (GEq v,GEq qv,GEq fun,GEq con,GEq field,GEq fv,GEq lv,GEq e)
=> GEq (Expression v qv fun con field fv lv e) where
geq (Var v1) (Var v2) = geq v1 v2
geq (QVar v1) (QVar v2) = geq v1 v2
geq (FVar v1) (FVar v2) = geq v1 v2
geq (LVar v1) (LVar v2) = geq v1 v2
geq (App f1 arg1) (App f2 arg2) = do
Refl <- geq f1 f2
Refl <- geq arg1 arg2
return Refl
geq (Const x) (Const y) = geq x y
geq (AsArray f1) (AsArray f2) = do
Refl <- geq f1 f2
return Refl
geq (Quantification q1 arg1 body1) (Quantification q2 arg2 body2)
| q1==q2 = do
Refl <- geq arg1 arg2
geq body1 body2
| otherwise = Nothing
geq (Let bnd1 body1) (Let bnd2 body2) = do
Refl <- geq bnd1 bnd2
geq body1 body2
geq _ _ = Nothing
instance (GEq v,GEq qv,GEq fun,GEq con,GEq field,GEq fv,GEq lv,GEq e)
=> Eq (Expression v qv fun con field fv lv e t) where
(==) = defaultEq
instance (GCompare v,GCompare qv,GCompare fun,GCompare con,
GCompare field,GCompare fv,GCompare lv,GCompare e)
=> GCompare (Expression v qv fun con field fv lv e) where
gcompare (Var v1) (Var v2) = gcompare v1 v2
gcompare (Var _) _ = GLT
gcompare _ (Var _) = GGT
gcompare (QVar v1) (QVar v2) = gcompare v1 v2
gcompare (QVar _) _ = GLT
gcompare _ (QVar _) = GGT
gcompare (FVar v1) (FVar v2) = gcompare v1 v2
gcompare (FVar _) _ = GLT
gcompare _ (FVar _) = GGT
gcompare (LVar v1) (LVar v2) = gcompare v1 v2
gcompare (LVar _) _ = GLT
gcompare _ (LVar _) = GGT
gcompare (App f1 arg1) (App f2 arg2) = case gcompare f1 f2 of
GEQ -> case gcompare arg1 arg2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (App _ _) _ = GLT
gcompare _ (App _ _) = GGT
gcompare (Const v1) (Const v2) = gcompare v1 v2
gcompare (Const _) _ = GLT
gcompare _ (Const _) = GGT
gcompare (AsArray f1) (AsArray f2) = case gcompare f1 f2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
gcompare (AsArray _) _ = GLT
gcompare _ (AsArray _) = GGT
gcompare (Quantification q1 arg1 body1) (Quantification q2 arg2 body2) = case compare q1 q2 of
LT -> GLT
GT -> GGT
EQ -> case gcompare arg1 arg2 of
GEQ -> gcompare body1 body2
GLT -> GLT
GGT -> GGT
gcompare (Quantification _ _ _) _ = GLT
gcompare _ (Quantification _ _ _) = GGT
gcompare (Let bnd1 body1) (Let bnd2 body2) = case gcompare bnd1 bnd2 of
GEQ -> gcompare body1 body2
GLT -> GLT
GGT -> GGT
instance (GCompare v,GCompare qv,GCompare fun,GCompare con,
GCompare field,GCompare fv,GCompare lv,GCompare e)
=> Ord (Expression v qv fun con field fv lv e t) where
compare = defaultCompare
instance (GEq fun,GEq con,GEq field) => GEq (Function fun con field) where
geq (Fun f1) (Fun f2) = geq f1 f2
geq (Eq tp1 n1) (Eq tp2 n2) = do
Refl <- geq tp1 tp2
Refl <- geq n1 n2
return Refl
geq (Distinct tp1 n1) (Distinct tp2 n2) = do
Refl <- geq tp1 tp2
Refl <- geq n1 n2
return Refl
geq (Map i1 f1) (Map i2 f2) = do
Refl <- geq f1 f2
Refl <- geq i1 i2
return Refl
geq (Ord tp1 o1) (Ord tp2 o2) = do
Refl <- geq tp1 tp2
if o1==o2 then return Refl else Nothing
geq (Arith tp1 o1 n1) (Arith tp2 o2 n2) = do
Refl <- geq tp1 tp2
if o1==o2
then do
Refl <- geq n1 n2
return Refl
else Nothing
geq (ArithIntBin o1) (ArithIntBin o2) = if o1==o2 then Just Refl else Nothing
geq Divide Divide = Just Refl
geq (Abs tp1) (Abs tp2) = do
Refl <- geq tp1 tp2
return Refl
geq Not Not = Just Refl
geq (Logic o1 n1) (Logic o2 n2)
= if o1==o2
then do
Refl <- geq n1 n2
return Refl
else Nothing
geq ToReal ToReal = Just Refl
geq ToInt ToInt = Just Refl
geq (ITE t1) (ITE t2) = do
Refl <- geq t1 t2
return Refl
geq (BVComp o1 bw1) (BVComp o2 bw2)
= if o1==o2
then do
Refl <- geq bw1 bw2
return Refl
else Nothing
geq (BVBin o1 bw1) (BVBin o2 bw2)
= if o1==o2
then do
Refl <- geq bw1 bw2
return Refl
else Nothing
geq (BVUn o1 bw1) (BVUn o2 bw2)
= if o1==o2
then do
Refl <- geq bw1 bw2
return Refl
else Nothing
geq (Select i1 e1) (Select i2 e2) = do
Refl <- geq i1 i2
Refl <- geq e1 e2
return Refl
geq (Store i1 e1) (Store i2 e2) = do
Refl <- geq i1 i2
Refl <- geq e1 e2
return Refl
geq (ConstArray i1 e1) (ConstArray i2 e2) = do
Refl <- geq i1 i2
Refl <- geq e1 e2
return Refl
geq (Concat a1 b1) (Concat a2 b2) = do
Refl <- geq a1 a2
Refl <- geq b1 b2
return Refl
geq (Extract bw1 start1 len1) (Extract bw2 start2 len2) = do
Refl <- geq bw1 bw2
Refl <- geq start1 start2
Refl <- geq len1 len2
return Refl
geq (Constructor c1) (Constructor c2) = do
Refl <- geq c1 c2
return Refl
geq (Test c1) (Test c2) = do
Refl <- geq c1 c2
return Refl
geq (Field f1) (Field f2) = do
Refl <- geq f1 f2
return Refl
geq (Divisible n1) (Divisible n2) = if n1==n2 then Just Refl else Nothing
geq _ _ = Nothing
instance (GCompare fun,GCompare con,GCompare field)
=> GCompare (Function fun con field) where
gcompare (Fun x) (Fun y) = gcompare x y
gcompare (Fun _) _ = GLT
gcompare _ (Fun _) = GGT
gcompare (Eq t1 n1) (Eq t2 n2) = case gcompare t1 t2 of
GEQ -> case gcompare n1 n2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Eq _ _) _ = GLT
gcompare _ (Eq _ _) = GGT
gcompare (Distinct t1 n1) (Distinct t2 n2) = case gcompare t1 t2 of
GEQ -> case gcompare n1 n2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Distinct _ _) _ = GLT
gcompare _ (Distinct _ _) = GGT
gcompare (Map i1 f1) (Map i2 f2) = case gcompare f1 f2 of
GEQ -> case gcompare i1 i2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Map _ _) _ = GLT
gcompare _ (Map _ _) = GGT
gcompare (Ord tp1 o1) (Ord tp2 o2) = case gcompare tp1 tp2 of
GEQ -> case compare o1 o2 of
EQ -> GEQ
LT -> GLT
GT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Ord _ _) _ = GLT
gcompare _ (Ord _ _) = GGT
gcompare (Arith tp1 o1 n1) (Arith tp2 o2 n2) = case gcompare tp1 tp2 of
GEQ -> case compare o1 o2 of
EQ -> case gcompare n1 n2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
LT -> GLT
GT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Arith _ _ _) _ = GLT
gcompare _ (Arith _ _ _) = GGT
gcompare (ArithIntBin o1) (ArithIntBin o2) = case compare o1 o2 of
EQ -> GEQ
LT -> GLT
GT -> GGT
gcompare (ArithIntBin _) _ = GLT
gcompare _ (ArithIntBin _) = GGT
gcompare Divide Divide = GEQ
gcompare Divide _ = GLT
gcompare _ Divide = GGT
gcompare (Abs tp1) (Abs tp2) = case gcompare tp1 tp2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
gcompare (Abs _) _ = GLT
gcompare _ (Abs _) = GGT
gcompare Not Not = GEQ
gcompare Not _ = GLT
gcompare _ Not = GGT
gcompare (Logic o1 n1) (Logic o2 n2) = case compare o1 o2 of
EQ -> case gcompare n1 n2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
LT -> GLT
GT -> GGT
gcompare (Logic _ _) _ = GLT
gcompare _ (Logic _ _) = GGT
gcompare ToReal ToReal = GEQ
gcompare ToReal _ = GLT
gcompare _ ToReal = GGT
gcompare ToInt ToInt = GEQ
gcompare ToInt _ = GLT
gcompare _ ToInt = GGT
gcompare (ITE t1) (ITE t2) = case gcompare t1 t2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
gcompare (ITE _) _ = GLT
gcompare _ (ITE _) = GGT
gcompare (BVComp o1 bw1) (BVComp o2 bw2) = case compare o1 o2 of
EQ -> case gcompare bw1 bw2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
LT -> GLT
GT -> GGT
gcompare (BVComp _ _) _ = GLT
gcompare _ (BVComp _ _) = GGT
gcompare (BVBin o1 bw1) (BVBin o2 bw2) = case compare o1 o2 of
EQ -> case gcompare bw1 bw2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
LT -> GLT
GT -> GGT
gcompare (BVBin _ _) _ = GLT
gcompare _ (BVBin _ _) = GGT
gcompare (BVUn o1 bw1) (BVUn o2 bw2) = case compare o1 o2 of
EQ -> case gcompare bw1 bw2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
LT -> GLT
GT -> GGT
gcompare (BVUn _ _) _ = GLT
gcompare _ (BVUn _ _) = GGT
gcompare (Select i1 e1) (Select i2 e2) = case gcompare i1 i2 of
GEQ -> case gcompare e1 e2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Select _ _) _ = GLT
gcompare _ (Select _ _) = GGT
gcompare (Store i1 e1) (Store i2 e2) = case gcompare i1 i2 of
GEQ -> case gcompare e1 e2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Store _ _) _ = GLT
gcompare _ (Store _ _) = GGT
gcompare (ConstArray i1 e1) (ConstArray i2 e2) = case gcompare i1 i2 of
GEQ -> case gcompare e1 e2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (ConstArray _ _) _ = GLT
gcompare _ (ConstArray _ _) = GGT
gcompare (Concat a1 b1) (Concat a2 b2) = case gcompare a1 a2 of
GEQ -> case gcompare b1 b2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Concat _ _) _ = GLT
gcompare _ (Concat _ _) = GGT
gcompare (Extract bw1 start1 len1) (Extract bw2 start2 len2)
= case gcompare bw1 bw2 of
GEQ -> case gcompare start1 start2 of
GEQ -> case gcompare len1 len2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
GLT -> GLT
GGT -> GGT
gcompare (Extract _ _ _) _ = GLT
gcompare _ (Extract _ _ _) = GGT
gcompare (Constructor c1) (Constructor c2) = case gcompare c1 c2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
gcompare (Constructor _) _ = GLT
gcompare _ (Constructor _) = GGT
gcompare (Test c1) (Test c2) = case gcompare c1 c2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
gcompare (Test _) _ = GLT
gcompare _ (Test _) = GGT
gcompare (Field f1) (Field f2) = case gcompare f1 f2 of
GEQ -> GEQ
GLT -> GLT
GGT -> GGT
gcompare (Field _) _ = GLT
gcompare _ (Field _) = GGT
gcompare (Divisible n1) (Divisible n2) = case compare n1 n2 of
EQ -> GEQ
LT -> GLT
GT -> GGT
data NoVar (t::Type) = NoVar'
data NoFun (sig::([Type],Type)) = NoFun'
data NoCon (sig::([Type],*)) = NoCon'
data NoField (sig::(*,Type)) = NoField'
instance GEq NoVar where
geq _ _ = error "geq for NoVar"
instance GEq NoFun where
geq _ _ = error "geq for NoFun"
instance GEq NoCon where
geq _ _ = error "geq for NoCon"
instance GEq NoField where
geq _ _ = error "geq for NoField"
instance GCompare NoVar where
gcompare _ _ = error "gcompare for NoVar"
instance GCompare NoFun where
gcompare _ _ = error "gcompare for NoFun"
instance GCompare NoCon where
gcompare _ _ = error "gcompare for NoCon"
instance GCompare NoField where
gcompare _ _ = error "gcompare for NoField"
instance Eq (NoVar t) where
(==) _ _ = error "== for NoVar"
instance Eq (NoFun t) where
(==) _ _ = error "== for NoFun"
instance Eq (NoCon t) where
(==) _ _ = error "== for NoCon"
instance Eq (NoField t) where
(==) _ _ = error "== for NoField"
instance Ord (NoVar t) where
compare _ _ = error "compare for NoVar"
instance Ord (NoFun t) where
compare _ _ = error "compare for NoFun"
instance Ord (NoCon t) where
compare _ _ = error "compare for NoCon"
instance Ord (NoField t) where
compare _ _ = error "compare for NoField"
instance Show (NoVar t) where
showsPrec _ _ = showString "NoVar"
instance GShow NoVar where
gshowsPrec = showsPrec
instance Show (NoFun t) where
showsPrec _ _ = showString "NoFun"
instance GShow NoFun where
gshowsPrec = showsPrec
instance Show (NoCon t) where
showsPrec _ _ = showString "NoCon"
instance GShow NoCon where
gshowsPrec = showsPrec
instance Show (NoField t) where
showsPrec _ _ = showString "NoVar"
instance GShow NoField where
gshowsPrec = showsPrec
instance GetType NoVar where
getType _ = error "getType called on NoVar."
instance GetFunType NoFun where
getFunType _ = error "getFunType called on NoFun."
instance GetConType NoCon where
getConType _ = error "getConType called on NoCon."
instance GetFieldType NoField where
getFieldType _ = error "getFieldType called on NoField."
instance (GetType v,GetType qv,GetFunType fun,GetConType con,
GetFieldType field,GetType fv,GetType lv,GetType e)
=> GetType (Expression v qv fun con field fv lv e) where
getType = runIdentity . expressionType
(return.getType) (return.getType) (return.getFunType) (return.getConType) (return.getFieldType)
(return.getType) (return.getType) (return.getType)
instance (GetFunType fun,GetConType con,GetFieldType field)
=> GetFunType (Function fun con field) where
getFunType = runIdentity . functionType
(return.getFunType) (return.getConType) (return.getFieldType)