{-| Copyright : (C) 2013-2016, University of Twente License : BSD2 (see the file LICENSE) Maintainer : Christiaan Baaij The Product/Signal isomorphism -} {-# LANGUAGE DataKinds #-} {-# LANGUAGE DefaultSignatures #-} {-# LANGUAGE KindSignatures #-} {-# LANGUAGE MagicHash #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE Trustworthy #-} {-# OPTIONS_HADDOCK show-extensions #-} module CLaSH.Signal.Bundle ( Bundle (..) ) where import Control.Applicative (liftA2) import GHC.TypeLits (KnownNat) import Prelude hiding (head, map, tail) import CLaSH.Signal.Internal (Clock, Signal' (..), SClock) import CLaSH.Sized.BitVector (BitVector) import CLaSH.Sized.Fixed (Fixed) import CLaSH.Sized.Index (Index) import CLaSH.Sized.Signed (Signed) import CLaSH.Sized.Unsigned (Unsigned) import CLaSH.Sized.Vector (Vec, traverse#) -- | Isomorphism between a 'CLaSH.Signal.Signal' of a product type (e.g. a tuple) and a -- product type of 'CLaSH.Signal.Signal''s. -- -- Instances of 'Bundle' must satisfy the following laws: -- -- @ -- 'bundle'' . 'unbundle'' = 'id' -- 'unbundle'' . 'bundle'' = 'id' -- @ -- -- By default, 'bundle'' and 'unbundle'', are defined as the identity, that is, -- writing: -- -- @ -- data D = A | B -- -- instance 'Bundle' D -- @ -- -- is the same as: -- -- @ -- data D = A | B -- -- instance 'Bundle' D where -- type 'Unbundled'' clk D = 'Signal'' clk D -- 'bundle'' _ s = s -- 'unbundle'' _ s = s -- @ -- class Bundle a where type Unbundled' (clk :: Clock) a type Unbundled' clk a = Signal' clk a -- | Example: -- -- @ -- __bundle'__ :: ('Signal'' clk a, 'Signal'' clk b) -> 'Signal'' clk (a,b) -- @ -- -- However: -- -- @ -- __bundle'__ :: 'Signal'' clk 'CLaSH.Sized.BitVector.Bit' -> 'Signal'' clk 'CLaSH.Sized.BitVector.Bit' -- @ bundle' :: SClock clk -> Unbundled' clk a -> Signal' clk a {-# INLINE bundle' #-} default bundle' :: SClock clk -> Signal' clk a -> Signal' clk a bundle' _ s = s -- | Example: -- -- @ -- __unbundle'__ :: 'Signal'' clk (a,b) -> ('Signal'' clk a, 'Signal'' clk b) -- @ -- -- However: -- -- @ -- __unbundle'__ :: 'Signal'' clk 'CLaSH.Sized.BitVector.Bit' -> 'Signal'' clk 'CLaSH.Sized.BitVector.Bit' -- @ unbundle' :: SClock clk -> Signal' clk a -> Unbundled' clk a {-# INLINE unbundle' #-} default unbundle' :: SClock clk -> Signal' clk a -> Signal' clk a unbundle' _ s = s instance Bundle Bool instance Bundle Integer instance Bundle Int instance Bundle Float instance Bundle Double instance Bundle () instance Bundle (Maybe a) instance Bundle (Either a b) instance Bundle (BitVector n) instance Bundle (Index n) instance Bundle (Fixed rep int frac) instance Bundle (Signed n) instance Bundle (Unsigned n) instance Bundle (a,b) where type Unbundled' t (a,b) = (Signal' t a, Signal' t b) bundle' _ = uncurry (liftA2 (,)) unbundle' _ tup = (fmap fst tup, fmap snd tup) instance Bundle (a,b,c) where type Unbundled' t (a,b,c) = (Signal' t a, Signal' t b, Signal' t c) bundle' _ (a,b,c) = (,,) <$> a <*> b <*> c unbundle' _ tup = (fmap (\(x,_,_) -> x) tup ,fmap (\(_,x,_) -> x) tup ,fmap (\(_,_,x) -> x) tup ) instance Bundle (a,b,c,d) where type Unbundled' t (a,b,c,d) = ( Signal' t a, Signal' t b, Signal' t c , Signal' t d ) bundle' _ (a,b,c,d) = (,,,) <$> a <*> b <*> c <*> d unbundle' _ tup = (fmap (\(x,_,_,_) -> x) tup ,fmap (\(_,x,_,_) -> x) tup ,fmap (\(_,_,x,_) -> x) tup ,fmap (\(_,_,_,x) -> x) tup ) instance Bundle (a,b,c,d,e) where type Unbundled' t (a,b,c,d,e) = ( Signal' t a, Signal' t b, Signal' t c , Signal' t d, Signal' t e ) bundle' _ (a,b,c,d,e) = (,,,,) <$> a <*> b <*> c <*> d <*> e unbundle' _ tup = (fmap (\(x,_,_,_,_) -> x) tup ,fmap (\(_,x,_,_,_) -> x) tup ,fmap (\(_,_,x,_,_) -> x) tup ,fmap (\(_,_,_,x,_) -> x) tup ,fmap (\(_,_,_,_,x) -> x) tup ) instance Bundle (a,b,c,d,e,f) where type Unbundled' t (a,b,c,d,e,f) = ( Signal' t a, Signal' t b, Signal' t c , Signal' t d, Signal' t e, Signal' t f ) bundle' _ (a,b,c,d,e,f) = (,,,,,) <$> a <*> b <*> c <*> d <*> e <*> f unbundle' _ 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 Bundle (a,b,c,d,e,f,g) where type Unbundled' t (a,b,c,d,e,f,g) = ( Signal' t a, Signal' t b, Signal' t c , Signal' t d, Signal' t e, Signal' t f , Signal' t g ) bundle' _ (a,b,c,d,e,f,g) = (,,,,,,) <$> a <*> b <*> c <*> d <*> e <*> f <*> g unbundle' _ 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 Bundle (a,b,c,d,e,f,g,h) where type Unbundled' t (a,b,c,d,e,f,g,h) = ( Signal' t a, Signal' t b, Signal' t c , Signal' t d, Signal' t e, Signal' t f , Signal' t g, Signal' t h ) bundle' _ (a,b,c,d,e,f,g,h) = (,,,,,,,) <$> a <*> b <*> c <*> d <*> e <*> f <*> g <*> h unbundle' _ 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 KnownNat n => Bundle (Vec n a) where type Unbundled' t (Vec n a) = Vec n (Signal' t a) -- The 'Traversable' instance of 'Vec' is not synthesisable, so we must -- define 'bundle'' as a primitive. bundle' = vecBundle# unbundle' _ = sequenceA {-# NOINLINE vecBundle# #-} vecBundle# :: SClock t -> Vec n (Signal' t a) -> Signal' t (Vec n a) vecBundle# _ = traverse# id