module Compose where data ST s = ST {runState :: s -> s} {-@ data ST s

Bool, q :: s -> s -> Bool> = ST (runState :: x:s

-> s) @-} {-@ runState :: forall

Bool, q :: s -> s -> Bool>. ST s -> x:s

-> s @-} {-@ cmp :: forall < pref :: s -> Bool, postf :: s -> s -> Bool , pre :: s -> Bool, postg :: s -> s -> Bool , post :: s -> s -> Bool >. {xx::s

, w::s |- s <: s}
       {ww::s
 |- s <: s}
       (ST  s)
    -> (ST  s)
    -> (ST  s)
@-}

cmp :: (ST s)
    -> (ST s)
    -> (ST s)

cmp (ST f) (ST g) = ST (\x -> f (g x))



{-@ incr :: ST <{\x -> x >= 0}, {\x v -> v = x + 1}>  Nat   @-}
incr :: ST Int
incr = ST $ \x ->  x + 1


{-@ incr2 :: ST <{\x -> x >= 0}, {\x v -> v = x + 5}>  Nat  @-}
incr2 :: ST Int
incr2 = cmp incr incr

{-@ incr3 :: ST <{\x -> x >= 0}, {\x v -> v = x + 4}>  Nat  @-}
incr3 :: ST Int
incr3 = cmp (cmp incr incr) incr


foo :: Int
{-@ foo :: {v:Nat |  v = 10} @-}
foo = runState incr3 0