Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Strongly separate clash-protocols-base and clash-protocols #118

Merged
merged 6 commits into from
Oct 3, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -118,7 +118,7 @@ jobs:
version: "0.14.0.0"
pattern: |
**/*.hs
!clash-protocols-base/src/Protocols/Cpp.hs
!clash-protocols-base/src/Protocols/Plugin/Cpp.hs

linting:
name: Source code linting
Expand Down
17 changes: 9 additions & 8 deletions clash-protocols-base/clash-protocols-base.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -111,15 +111,16 @@ library
, template-haskell

exposed-modules:
Protocols.Cpp
Protocols.Internal
Protocols.Internal.Classes
Protocols.Internal.TaggedBundle
Protocols.Internal.TaggedBundle.TH
Protocols.Internal.TH
Protocols.Internal.Units
Protocols.Internal.Units.TH
Protocols.Plugin
Protocols.Plugin.Cpp
Protocols.Plugin.Internal
Protocols.Plugin.TaggedBundle
Protocols.Plugin.TaggedBundle.TH
Protocols.Plugin.TH
Protocols.Plugin.Units
Protocols.Plugin.Units.TH

other-modules:
Protocols.Plugin.Types

default-language: Haskell2010
47 changes: 42 additions & 5 deletions clash-protocols-base/src/Protocols/Plugin.hs
Original file line number Diff line number Diff line change
@@ -1,11 +1,18 @@
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE PatternSynonyms #-}
{-# OPTIONS_GHC -fno-warn-orphans #-}

{- |
A GHC source plugin providing a DSL for writing Circuit components. Credits to
@circuit-notation@ at <https://github.com/cchalmers/circuit-notation>.
-}
module Protocols.Plugin (
-- * Circuit types
Circuit (..),
Protocol (..),

-- * clash-prelude related types
CSignal,

-- * plugin functions
plugin,
circuit,
(-<),
Expand All @@ -14,11 +21,16 @@ module Protocols.Plugin (
-- base
import Prelude

-- clash-prelude
import qualified Clash.Explicit.Prelude as C

-- clash-protocols
import Protocols.Internal
import Protocols.Internal.TaggedBundle
import Protocols.Internal.Units
import Protocols.Plugin.Cpp
import Protocols.Plugin.Internal
import Protocols.Plugin.TH
import Protocols.Plugin.TaggedBundle
import Protocols.Plugin.Types
import Protocols.Plugin.Units

-- circuit-notation
import qualified CircuitNotation as CN
Expand All @@ -29,6 +41,31 @@ import Data.Tagged
-- ghc
import qualified GHC.Plugins as GHC

instance Protocol () where
type Fwd () = ()
type Bwd () = ()

{- | __NB__: The documentation only shows instances up to /3/-tuples. By
default, instances up to and including /12/-tuples will exist. If the flag
@large-tuples@ is set instances up to the GHC imposed limit will exist. The
GHC imposed limit is either 62 or 64 depending on the GHC version.
-}
instance Protocol (a, b) where
Comment on lines +48 to +52
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is slightly misleading. People will encounter this documentation in clash-protocols. So they might set the -flarge-tuples flag on clash-protocols when really they should set it on both clash-protocols-base as well as clash-protocols.

However, I now see it's actually worse than that. The instances defined in clash-protocols-base are not documented at all in the Haddock for clash-protocols. I think this is a real shortcoming. People will not easily be able to know that these instances exist at all.

type Fwd (a, b) = (Fwd a, Fwd b)
type Bwd (a, b) = (Bwd a, Bwd b)

-- Generate n-tuple instances, where n > 2
protocolTupleInstances 3 maxTupleSize

instance (C.KnownNat n) => Protocol (C.Vec n a) where
type Fwd (C.Vec n a) = C.Vec n (Fwd a)
type Bwd (C.Vec n a) = C.Vec n (Bwd a)

-- XXX: Type families with Signals on LHS are currently broken on Clash:
instance Protocol (CSignal dom a) where
type Fwd (CSignal dom a) = C.Signal dom a
type Bwd (CSignal dom a) = C.Signal dom ()

-- | @circuit-notation@ plugin repurposed for "Protocols".
plugin :: GHC.Plugin
plugin =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ Compile-time dependent constants. Inspired by @clash-prelude@'s @Clash.CPP@.

{-# OPTIONS_HADDOCK hide #-}

module Protocols.Cpp
module Protocols.Plugin.Cpp
( maxTupleSize
, haddockOnly
) where
Expand Down
17 changes: 16 additions & 1 deletion clash-protocols-base/src/Protocols/Plugin/Internal.hs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,22 @@ module Protocols.Plugin.Internal where
import Clash.Explicit.Prelude

import Data.Tagged
import Protocols.Internal
import GHC.Base (Any)
import Protocols.Plugin.Types

{- | Picked up by "Protocols.Plugin" to process protocol DSL. See
"Protocols.Plugin" for more information.
-}
circuit :: Any
circuit =
error "'circuit' called: did you forget to enable \"Protocols.Plugin\"?"

{- | Picked up by "Protocols.Plugin" to tie circuits together. See
"Protocols.Plugin" for more information.
-}
(-<) :: Any
(-<) =
error "(-<) called: did you forget to enable \"Protocols.Plugin\"?"

{- | Convenience type alias. A circuit where all parts are decorated with a
tag, referring to the @a@ and @b@ in its main signature. This is (indirectly)
Expand Down
36 changes: 36 additions & 0 deletions clash-protocols-base/src/Protocols/Plugin/TH.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
{-# OPTIONS_HADDOCK hide #-}
DigitalBrains1 marked this conversation as resolved.
Show resolved Hide resolved

module Protocols.Plugin.TH where

import Language.Haskell.TH

appTs :: Q Type -> [Q Type] -> Q Type
appTs = foldl appT

-- | Generate @Protocol@ instances for n-tuples
protocolTupleInstances :: Int -> Int -> Q [Dec]
protocolTupleInstances n m = mapM protocolTupleInstance [n .. m]

protocolTupleInstance :: Int -> Q Dec
protocolTupleInstance n =
instanceD
(pure []) -- context
(protocolConT `appT` tup) -- head
[mkTyInst fwdConName, mkTyInst bwdConName] -- body
where
fwdConName = mkName "Fwd"
bwdConName = mkName "Bwd"
protocolConT = conT (mkName "Protocol")

tyVars :: [TypeQ]
tyVars = map (varT . mkName . ('a' :) . show) [1 .. n]

tup = tupleT n `appTs` tyVars

mkTyInst :: Name -> DecQ
mkTyInst con =
tySynInstD $ tySynEqn Nothing lhs rhs
where
lhs, rhs :: TypeQ
lhs = conT con `appT` tup
rhs = tupleT n `appTs` map (conT con `appT`) tyVars
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,12 @@
-- For debugging TH:
-- {-# OPTIONS_GHC -ddump-splices #-}

module Protocols.Internal.TaggedBundle where
module Protocols.Plugin.TaggedBundle where

import Clash.Explicit.Prelude

import Protocols.Cpp (maxTupleSize)
import Protocols.Internal.TaggedBundle.TH (taggedBundleTupleInstances)
import Protocols.Plugin.Cpp (maxTupleSize)
import Protocols.Plugin.TaggedBundle.TH (taggedBundleTupleInstances)

import Data.Tagged

Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{-# OPTIONS_HADDOCK hide #-}

module Protocols.Internal.TaggedBundle.TH where
module Protocols.Plugin.TaggedBundle.TH where

import Data.Tagged
import Language.Haskell.TH
Expand Down
Original file line number Diff line number Diff line change
@@ -1,15 +1,17 @@
{-# LANGUAGE RoleAnnotations #-}
{-# OPTIONS_HADDOCK hide #-}

{- |
These class definitions are needed to be able to write Template Haskell quotes
for instances. They are defined separately to avoid import loops.

This module is not exported; the classes and their (orphan) instances are
exported elsewhere.
-}
module Protocols.Internal.Classes where
module Protocols.Plugin.Types where

import Clash.Signal
import Data.Kind (Type)
import Data.Proxy

-- | A protocol describes the in- and outputs of one side of a 'Circuit'.
class Protocol a where
Expand Down Expand Up @@ -134,36 +136,12 @@ types:
newtype Circuit a b
= Circuit ((Fwd a, Bwd b) -> (Bwd a, Fwd b))

{- | Idle state of a Circuit. Aims to provide no data for both the forward and
backward direction. Transactions are not acknowledged.
{- | Circuit protocol with /Signal dom a/ in its forward direction, and
/()/ in its backward direction. Convenient for exposing protocol
internals, or simply for undirectional streams.
Note: 'CSignal' exists to work around [issue 760](https://github.com/clash-lang/clash-compiler/issues/760)
in Clash, where type families with 'Signal' on the LHS are broken.
-}
class (Protocol p) => IdleCircuit p where
idleFwd :: Proxy p -> Fwd (p :: Type)
idleBwd :: Proxy p -> Bwd (p :: Type)
data CSignal (dom :: Domain) (a :: Type)

{- | Force a /nack/ on the backward channel and /no data/ on the forward
channel if reset is asserted.
-}
forceResetSanityGeneric ::
forall dom a fwd bwd.
( KnownDomain dom
, HiddenReset dom
, IdleCircuit a
, Fwd a ~ Signal dom fwd
, Bwd a ~ Signal dom bwd
) =>
Circuit a a
forceResetSanityGeneric = Circuit go
where
go (fwd, bwd) =
unbundle $
mux
rstAsserted
(bundle (idleBwd $ Proxy @a, idleFwd $ Proxy @a))
(bundle (bwd, fwd))

#if MIN_VERSION_clash_prelude(1,8,0)
rstAsserted = unsafeToActiveHigh hasReset
#else
rstAsserted = unsafeToHighPolarity hasReset
#endif
type role CSignal nominal representational
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,12 @@
-- For debugging TH:
-- {-# OPTIONS_GHC -ddump-splices #-}

module Protocols.Internal.Units where
module Protocols.Plugin.Units where

import Clash.Explicit.Prelude

import Protocols.Cpp (maxTupleSize)
import Protocols.Internal.Units.TH (unitsTupleInstances)
import Protocols.Plugin.Cpp (maxTupleSize)
import Protocols.Plugin.Units.TH (unitsTupleInstances)

{- | Utilities for zero-width types. Is used by "Protocols.Plugin" to drive \"trivial\"
backwards channels.
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{-# OPTIONS_HADDOCK hide #-}

module Protocols.Internal.Units.TH (unitsTupleInstances) where
module Protocols.Plugin.Units.TH (unitsTupleInstances) where

import Language.Haskell.TH

Expand Down
3 changes: 3 additions & 0 deletions clash-protocols/clash-protocols.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -145,10 +145,12 @@ library
Protocols.Axi4.WriteData
Protocols.Axi4.WriteResponse
Protocols.Df
Protocols.Internal
Protocols.DfConv
Protocols.Hedgehog
Protocols.Hedgehog.Internal
Protocols.Idle
Protocols.Internal.TH
Protocols.Wishbone
Protocols.Wishbone.Standard
Protocols.Wishbone.Standard.Hedgehog
Expand All @@ -163,6 +165,7 @@ library
autogen-modules: Paths_clash_protocols

other-modules:
Protocols.Internal.Types
Paths_clash_protocols

default-language: Haskell2010
Expand Down
6 changes: 2 additions & 4 deletions clash-protocols/src/Protocols.hs
Original file line number Diff line number Diff line change
Expand Up @@ -59,12 +59,10 @@ module Protocols (
-- * Circuit notation plugin
circuit,
(-<),
module Protocols.Internal.Units,
module Protocols.Internal.TaggedBundle,
Units (..),
TaggedBundle (..),
) where

import Data.Default (def)
import Protocols.Df (Df)
import Protocols.Internal
import Protocols.Internal.TaggedBundle
import Protocols.Internal.Units
1 change: 1 addition & 0 deletions clash-protocols/src/Protocols/Axi4/ReadAddress.hs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@

-- me
import Protocols.Axi4.Common
import Protocols.Idle
import Protocols.Internal

-- | Configuration options for 'Axi4ReadAddress'.
Expand Down Expand Up @@ -188,7 +189,7 @@
-- | See Table A2-5 "Read address channel signals"
newtype S2M_ReadAddress = S2M_ReadAddress
{_arready :: Bool}
deriving (Show, Generic, C.NFDataX)

Check warning on line 192 in clash-protocols/src/Protocols/Axi4/ReadAddress.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.2.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 192 in clash-protocols/src/Protocols/Axi4/ReadAddress.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.4.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 192 in clash-protocols/src/Protocols/Axi4/ReadAddress.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.6.4 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

{- | Shorthand for a "well-behaved" read address config,
so that we don't need to write out a bunch of type constraints later.
Expand Down
1 change: 1 addition & 0 deletions clash-protocols/src/Protocols/Axi4/Stream.hs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
import qualified Protocols.Df as Df
import qualified Protocols.DfConv as DfConv
import Protocols.Hedgehog.Internal
import Protocols.Idle
import Protocols.Internal

instance (KnownNat n) => Hashable (Unsigned n)
Expand Down Expand Up @@ -102,7 +103,7 @@
the '_tready' signal.
-}
newtype Axi4StreamS2M = Axi4StreamS2M {_tready :: Bool}
deriving (Generic, C.NFDataX, C.ShowX, Eq, NFData, Show, Bundle)

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.2.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.2.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.2.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.4.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.4.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.4.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.6.4 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.6.4 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 106 in clash-protocols/src/Protocols/Axi4/Stream.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.6.4 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

-- | Type for AXI4 Stream protocol.
data Axi4Stream (dom :: Domain) (conf :: Axi4StreamConfig) (userType :: Type)
Expand Down
1 change: 1 addition & 0 deletions clash-protocols/src/Protocols/Axi4/WriteAddress.hs
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@

-- me
import Protocols.Axi4.Common
import Protocols.Idle
import Protocols.Internal

-- | Configuration options for 'Axi4WriteAddress'.
Expand Down Expand Up @@ -185,7 +186,7 @@

-- | See Table A2-2 "Write address channel signals"
newtype S2M_WriteAddress = S2M_WriteAddress {_awready :: Bool}
deriving (Show, Generic, C.NFDataX)

Check warning on line 189 in clash-protocols/src/Protocols/Axi4/WriteAddress.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.2.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 189 in clash-protocols/src/Protocols/Axi4/WriteAddress.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.4.8 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

Check warning on line 189 in clash-protocols/src/Protocols/Axi4/WriteAddress.hs

View workflow job for this annotation

GitHub Actions / Cabal tests - ghc 9.6.4 / clash 1.8.1

• Both DeriveAnyClass and GeneralizedNewtypeDeriving are enabled

{- | Shorthand for a "well-behaved" write address config,
so that we don't need to write out a bunch of type constraints later.
Expand Down
1 change: 1 addition & 0 deletions clash-protocols/src/Protocols/Df.hs
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,7 @@ import qualified Clash.Prelude as C
import Clash.Signal.Internal (Signal (..))

-- me
import Protocols.Idle
import Protocols.Internal

{-# ANN module "HLint: ignore Use const" #-}
Expand Down
Loading
Loading