module CLaSH.Signal
( Signal
, sample
, sampleN
, fromList
, signal
, register
, simulate
, Pack(..)
, simulateP
, (<^), (^>)
)
where
import Data.Default
import Control.Applicative
import Language.Haskell.TH.Syntax(Lift(..))
import CLaSH.Bit (Bit)
import CLaSH.Sized.Signed (Signed)
import CLaSH.Sized.Unsigned (Unsigned)
import CLaSH.Sized.Vector (Vec(..), vmap, vhead, vtail)
infixr 5 :-
data Signal a = a :- Signal a
fromList :: [a] -> Signal a
fromList [] = error "finite list"
fromList (x:xs) = x :- fromList xs
instance Show a => Show (Signal a) where
show (x :- xs) = show x ++ " " ++ show xs
instance Lift a => Lift (Signal a) where
lift ~(x :- _) = [| signal x |]
instance Default a => Default (Signal a) where
def = signal def
sample :: Signal a -> [a]
sample ~(x :- xs) = x : sample xs
sampleN :: Int -> Signal a -> [a]
sampleN 0 _ = []
sampleN n ~(x :- xs) = x : (sampleN (n1) xs)
signal :: a -> Signal a
signal a = a :- signal a
mapSignal :: (a -> b) -> Signal a -> Signal b
mapSignal f (a :- as) = f a :- mapSignal f as
appSignal :: Signal (a -> b) -> Signal a -> Signal b
appSignal (f :- fs) ~(a :- as) = f a :- appSignal fs as
instance Functor Signal where
fmap = mapSignal
instance Applicative Signal where
pure = signal
(<*>) = appSignal
unSignal :: Signal a -> a
unSignal (a :- _) = a
next :: Signal a -> Signal a
next (_ :- as) = as
diag :: Signal (Signal a) -> Signal a
diag (xs :- xss) = unSignal xs :- diag (fmap next xss)
instance Monad Signal where
return = signal
xs >>= f = diag (fmap f xs)
register :: a -> Signal a -> Signal a
register i s = i :- s
simulate :: (Signal a -> Signal b) -> [a] -> [b]
simulate f = sample . f . fromList
class Pack a where
type SignalP a
pack :: SignalP a -> Signal a
unpack :: Signal a -> SignalP a
simulateP :: (Pack a, Pack b) => (SignalP a -> SignalP b) -> [a] -> [b]
simulateP f = simulate (pack . f . unpack)
instance Pack Bit where
type SignalP Bit = Signal Bit
pack = id
unpack = id
instance Pack (Signed n) where
type SignalP (Signed n) = Signal (Signed n)
pack = id
unpack = id
instance Pack (Unsigned n) where
type SignalP (Unsigned n) = Signal (Unsigned n)
pack = id
unpack = id
instance Pack Bool where
type SignalP Bool = Signal Bool
pack = id
unpack = id
instance Pack Integer where
type SignalP Integer = Signal Integer
pack = id
unpack = id
instance Pack Int where
type SignalP Int = Signal Int
pack = id
unpack = id
instance Pack Float where
type SignalP Float = Signal Float
pack = id
unpack = id
instance Pack Double where
type SignalP Double = Signal Double
pack = id
unpack = id
instance Pack () where
type SignalP () = Signal ()
pack = id
unpack = id
instance Pack (a,b) where
type SignalP (a,b) = (Signal a, Signal b)
pack = uncurry (liftA2 (,))
unpack tup = (fmap fst tup, fmap snd tup)
instance Pack (a,b,c) where
type SignalP (a,b,c) = (Signal a, Signal b, Signal c)
pack (a,b,c) = (,,) <$> a <*> b <*> c
unpack tup = (fmap (\(x,_,_) -> x) tup
,fmap (\(_,x,_) -> x) tup
,fmap (\(_,_,x) -> x) tup
)
instance Pack (a,b,c,d) where
type SignalP (a,b,c,d) = (Signal a, Signal b, Signal c, Signal d)
pack (a,b,c,d) = (,,,) <$> a <*> b <*> c <*> d
unpack tup = (fmap (\(x,_,_,_) -> x) tup
,fmap (\(_,x,_,_) -> x) tup
,fmap (\(_,_,x,_) -> x) tup
,fmap (\(_,_,_,x) -> x) tup
)
instance Pack (a,b,c,d,e) where
type SignalP (a,b,c,d,e) = (Signal a, Signal b, Signal c, Signal d, Signal e)
pack (a,b,c,d,e) = (,,,,) <$> a <*> b <*> c <*> d <*> e
unpack tup = (fmap (\(x,_,_,_,_) -> x) tup
,fmap (\(_,x,_,_,_) -> x) tup
,fmap (\(_,_,x,_,_) -> x) tup
,fmap (\(_,_,_,x,_) -> x) tup
,fmap (\(_,_,_,_,x) -> x) tup
)
instance Pack (a,b,c,d,e,f) where
type SignalP (a,b,c,d,e,f) = (Signal a, Signal b, Signal c, Signal d, Signal e, Signal f)
pack (a,b,c,d,e,f) = (,,,,,) <$> a <*> b <*> c <*> d <*> e <*> f
unpack tup = (fmap (\(x,_,_,_,_,_) -> x) tup
,fmap (\(_,x,_,_,_,_) -> x) tup
,fmap (\(_,_,x,_,_,_) -> x) tup
,fmap (\(_,_,_,x,_,_) -> x) tup
,fmap (\(_,_,_,_,x,_) -> x) tup
,fmap (\(_,_,_,_,_,x) -> x) tup
)
instance Pack (a,b,c,d,e,f,g) where
type SignalP (a,b,c,d,e,f,g) = (Signal a, Signal b, Signal c, Signal d, Signal e, Signal f, Signal g)
pack (a,b,c,d,e,f,g) = (,,,,,,) <$> a <*> b <*> c <*> d <*> e <*> f <*> g
unpack tup = (fmap (\(x,_,_,_,_,_,_) -> x) tup
,fmap (\(_,x,_,_,_,_,_) -> x) tup
,fmap (\(_,_,x,_,_,_,_) -> x) tup
,fmap (\(_,_,_,x,_,_,_) -> x) tup
,fmap (\(_,_,_,_,x,_,_) -> x) tup
,fmap (\(_,_,_,_,_,x,_) -> x) tup
,fmap (\(_,_,_,_,_,_,x) -> x) tup
)
instance Pack (a,b,c,d,e,f,g,h) where
type SignalP (a,b,c,d,e,f,g,h) = (Signal a, Signal b, Signal c, Signal d, Signal e, Signal f, Signal g, Signal h)
pack (a,b,c,d,e,f,g,h) = (,,,,,,,) <$> a <*> b <*> c <*> d <*> e <*> f <*> g <*> h
unpack tup = (fmap (\(x,_,_,_,_,_,_,_) -> x) tup
,fmap (\(_,x,_,_,_,_,_,_) -> x) tup
,fmap (\(_,_,x,_,_,_,_,_) -> x) tup
,fmap (\(_,_,_,x,_,_,_,_) -> x) tup
,fmap (\(_,_,_,_,x,_,_,_) -> x) tup
,fmap (\(_,_,_,_,_,x,_,_) -> x) tup
,fmap (\(_,_,_,_,_,_,x,_) -> x) tup
,fmap (\(_,_,_,_,_,_,_,x) -> x) tup
)
instance Pack (Vec n a) where
type SignalP (Vec n a) = Vec n (Signal a)
pack vs = vmap unSignal vs :- pack (vmap next vs)
unpack (Nil :- _) = Nil
unpack vs@((_ :> _) :- _) = fmap vhead vs :> (unpack (fmap vtail vs))
(<^) :: Applicative f => f a -> (a -> b -> c) -> f b -> f c
v <^ f = liftA2 f v
(^>) :: Applicative f => (f a -> f b) -> f a -> f b
f ^> v = f v
instance Num a => Num (Signal a) where
(+) = liftA2 (+)
() = liftA2 ()
(*) = liftA2 (*)
negate = fmap negate
abs = fmap abs
signum = fmap signum
fromInteger = signal . fromInteger