Skip to content

Commit

Permalink
Add backPressureTupleInstances and drivableTupleInstances
Browse files Browse the repository at this point in the history
  • Loading branch information
lmbollen committed Sep 27, 2024
1 parent 80b24a2 commit 17e4ea5
Show file tree
Hide file tree
Showing 2 changed files with 75 additions and 12 deletions.
20 changes: 8 additions & 12 deletions clash-protocols-base/src/Protocols/Internal.hs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,12 @@ import qualified Clash.Prelude as C

import Protocols.Cpp (maxTupleSize)
import Protocols.Internal.Classes
import Protocols.Internal.TH (protocolTupleInstances, simulateTupleInstances)
import Protocols.Internal.TH (
backPressureTupleInstances,
drivableTupleInstances,
protocolTupleInstances,
simulateTupleInstances,
)

import Control.Arrow ((***))
import Data.Coerce (coerce)
Expand Down Expand Up @@ -126,13 +131,7 @@ instance Backpressure () where
instance (Backpressure a, Backpressure b) => Backpressure (a, b) where
boolsToBwd _ bs = (boolsToBwd (Proxy @a) bs, boolsToBwd (Proxy @b) bs)

instance (Backpressure a, Backpressure b, Backpressure c) => Backpressure (a, b, c) where
boolsToBwd _ bs =
( boolsToBwd (Proxy @a) bs
, boolsToBwd (Proxy @b) bs
, boolsToBwd (Proxy @c) bs
)

backPressureTupleInstances 3 maxTupleSize
instance (C.KnownNat n, Backpressure a) => Backpressure (C.Vec n a) where
boolsToBwd _ bs = C.repeat (boolsToBwd (Proxy @a) bs)

Expand Down Expand Up @@ -292,10 +291,7 @@ instance (Drivable a, Drivable b) => Drivable (a, b) where
, sampleC @b conf (Circuit $ \_ -> ((), fwd2))
)

-- TODO TemplateHaskell?
-- instance SimulateType (a, b, c)
-- instance SimulateType (a, b, c, d)

drivableTupleInstances 3 maxTupleSize
instance (CE.KnownNat n, Simulate a) => Simulate (C.Vec n a) where
type SimulateFwdType (C.Vec n a) = C.Vec n (SimulateFwdType a)
type SimulateBwdType (C.Vec n a) = C.Vec n (SimulateBwdType a)
Expand Down
67 changes: 67 additions & 0 deletions clash-protocols-base/src/Protocols/Internal/TH.hs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ module Protocols.Internal.TH where

import qualified Clash.Prelude as C
import Control.Monad.Extra (concatMapM)
import Data.Proxy
import GHC.TypeNats
import Language.Haskell.TH
import Protocols.Internal.Classes
Expand Down Expand Up @@ -137,3 +138,69 @@ simulateTupleInstance n =
tupE [varE $ mkName $ "fwd" <> show i, varE $ mkName $ "bwd" <> show i]
)
[]

drivableTupleInstances :: Int -> Int -> DecsQ
drivableTupleInstances n m = concatMapM drivableTupleInstance [n .. m]

drivableTupleInstance :: Int -> DecsQ
drivableTupleInstance n =
[d|
instance ($instCtx) => Drivable $instTy where
type
ExpectType $instTy =
$(foldl appT (tupleT n) $ map (\ty -> [t|ExpectType $ty|]) circTys)
toSimulateType Proxy $(tupP circPats) = $toSimulateExpr

fromSimulateType Proxy $(tupP circPats) = $fromSimulateExpr

driveC $(varP $ mkName "conf") $(tupP fwdPats) = $(letE driveCDecs driveCExpr)
sampleC conf (Circuit f) =
let
$(varP $ mkName "bools") = replicate (resetCycles conf) False <> repeat True
(_, $(tupP fwdPats)) = f ((), $(tupE $ map mkSampleCExpr circTys))
in
$( tupE $
zipWith (\ty fwd -> [|sampleC @($ty) conf (Circuit $ const ((), $fwd))|]) circTys fwdExprs
)
|]
where
circStrings = map (\i -> "c" <> show i) [1 .. n]
circTys = map (varT . mkName) circStrings
circPats = map (varP . mkName) circStrings
circExprs = map (varE . mkName) circStrings
instCtx = foldl appT (tupleT n) $ map (\ty -> [t|Drivable $ty|]) circTys
instTy = foldl appT (tupleT n) circTys
fwdPats = map (varP . mkName . ("fwd" <>)) circStrings
fwdExprs = map (varE . mkName . ("fwd" <>)) circStrings
bwdExprs = map (varE . mkName . ("bwd" <>)) circStrings
bwdPats = map (varP . mkName . ("bwd" <>)) circStrings

mkSampleCExpr ty = [e|boolsToBwd (Proxy @($ty)) bools|]
driveCDecs =
pure $
valD
(tupP $ map (\p -> [p|(Circuit $p)|]) circPats)
(normalB $ tupE $ zipWith (\ty fwd -> [e|driveC @($ty) conf $fwd|]) circTys fwdExprs)
[]

driveCExpr =
[e|
Circuit $ \(_, $(tildeP $ tupP bwdPats)) -> ((), $(tupE $ zipWith mkDriveCExpr circExprs bwdExprs))
|]
mkDriveCExpr c bwd = [e|snd ($c ((), $bwd))|]
toSimulateExpr = tupE $ zipWith (\ty c -> [|toSimulateType (Proxy @($ty)) $c|]) circTys circExprs
fromSimulateExpr = tupE $ zipWith (\ty c -> [|fromSimulateType (Proxy @($ty)) $c|]) circTys circExprs

backPressureTupleInstances :: Int -> Int -> DecsQ
backPressureTupleInstances n m = concatMapM backPressureTupleInstance [n .. m]

backPressureTupleInstance :: Int -> DecsQ
backPressureTupleInstance n =
[d|
instance ($instCtx) => Backpressure $instTy where
boolsToBwd _ bs = $(tupE $ map (\ty -> [e|boolsToBwd (Proxy @($ty)) bs|]) circTys)
|]
where
circTys = map (\i -> varT $ mkName $ "c" <> show i) [1 .. n]
instCtx = foldl appT (tupleT n) $ map (\ty -> [t|Backpressure $ty|]) circTys
instTy = foldl appT (tupleT n) circTys

0 comments on commit 17e4ea5

Please sign in to comment.