Skip to content

Commit

Permalink
Add tests for Finite laws
Browse files Browse the repository at this point in the history
  • Loading branch information
kleinreact committed Jan 10, 2025
1 parent 59b40a6 commit 645d9fa
Show file tree
Hide file tree
Showing 3 changed files with 250 additions and 0 deletions.
2 changes: 2 additions & 0 deletions clash-prelude/clash-prelude.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -412,6 +412,7 @@ test-suite unittests

base,
bytestring,
constraints >= 0.9 && < 1.0,
deepseq,
hedgehog >= 1.0.3 && < 1.5,
hint >= 0.7 && < 0.10,
Expand Down Expand Up @@ -451,6 +452,7 @@ test-suite unittests
Clash.Tests.XException

Clash.Tests.Laws.Enum
Clash.Tests.Laws.Finite
Clash.Tests.Laws.SaturatingNum

Hedgehog.Extra
Expand Down
246 changes: 246 additions & 0 deletions clash-prelude/tests/Clash/Tests/Laws/Finite.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,246 @@
{-|
Copyright : (C) 2024-2025, QBayLogic B.V.
License : BSD2 (see the file LICENSE)
Maintainer : QBayLogic B.V. <[email protected]>
-}

{-# LANGUAGE CPP #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE TypeFamilies #-}

module Clash.Tests.Laws.Finite (tests) where

import Prelude hiding (reverse)

import Control.DeepSeq (NFData)
import Control.Monad (forM_)
import Data.Constraint (Dict(..))
import Data.Functor.Compose (Compose(..))
import Data.Functor.Const (Const(..))
import Data.Functor.Identity (Identity(..))
import Data.Functor.Product (Product)
import Data.Functor.Sum (Sum)
import Data.Int (Int8, Int16)
import Data.Ord (Down(..))
import Data.Proxy (Proxy(..))
import Data.Typeable (Typeable, typeRep)
import Data.Void (Void)
import Data.Word (Word8, Word16)
import Test.Tasty (TestTree, testGroup)
import Test.Tasty.HUnit (Assertion, (@=?), testCase)

import Clash.Class.Finite (Finite(..))
import Clash.Promoted.Nat (SNatLE(..), SNat(..), compareSNat)
import Clash.Sized.BitVector (BitVector, Bit)
import Clash.Sized.Index (Index)
import Clash.Sized.RTree (RTree)
import Clash.Sized.Signed (Signed)
import Clash.Sized.Unsigned (Unsigned)
import Clash.Sized.Vector (Vec, indicesI, iterateI, reverse)

indexOrderLaw ::
forall a.
(NFData a, Show a, Finite a) =>
Proxy a ->
Assertion
indexOrderLaw Proxy =
index <$> elements @a @=? indicesI

forwardIterateLaw ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Proxy a ->
Assertion
forwardIterateLaw Proxy =
iterateI (>>= succMaybe) (lowestMaybe @a) @=? Just <$> elements @a

backwardIterateLaw ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Proxy a ->
Assertion
backwardIterateLaw Proxy =
iterateI (>>= predMaybe) (highestMaybe @a) @=? Just <$> reverse (elements @a)

indexIsomorphismLaw ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Proxy a ->
Assertion
indexIsomorphismLaw Proxy =
ith . index <$> elements @a @=? elements @a

minimumPredecessor ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Proxy a ->
Assertion
minimumPredecessor Proxy =
(lowestMaybe >>= predMaybe @a) @=? Nothing

maximumSuccessor ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Proxy a ->
Assertion
maximumSuccessor Proxy =
(highestMaybe >>= succMaybe @a) @=? Nothing

extremes ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Proxy a ->
Assertion
extremes Proxy = case compareSNat (SNat @1) (SNat @(ElementCount a)) of
SNatLE -> do
lowestMaybe @a @=? Just lowest
highestMaybe @a @=? Just highest
SNatGT -> do
lowestMaybe @a @=? Nothing
highestMaybe @a @=? Nothing

boundedCompatibility ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Maybe (Dict (Bounded a)) ->
Assertion
boundedCompatibility = \case
Nothing -> return ()
Just Dict -> case compareSNat (SNat @1) (SNat @(ElementCount a)) of
SNatGT -> return ()
SNatLE -> do
lowest @a @=? minBound @a
highest @a @=? maxBound @a

enumCompatibility ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Maybe (Dict (Enum a)) ->
Assertion
enumCompatibility = \case
Nothing -> return ()
Just Dict -> forM_ (elements @a) $ \x -> do
maybe (return ()) (@=? succ x) $ succMaybe x
maybe (return ()) (@=? pred x) $ predMaybe x

finiteLaws ::
forall a.
(NFData a, Show a, Eq a, Finite a) =>
Maybe (Dict (Bounded a)) ->
Maybe (Dict (Enum a)) ->
[TestTree]
finiteLaws mBounded mEnum =
[ testCase "Index Order" $ indexOrderLaw proxy
, testCase "Forward Iterate" $ forwardIterateLaw proxy
, testCase "Backward Iterate" $ backwardIterateLaw proxy
, testCase "Index Isomorphism" $ indexIsomorphismLaw proxy
, testCase "Minimum Predecessor" $ minimumPredecessor proxy
, testCase "Maximum Successor" $ maximumSuccessor proxy
, testCase "Extremes" $ extremes proxy
, testCase "Bounded Compatibility" $ boundedCompatibility mBounded
, testCase "Enum Compatibility" $ enumCompatibility mEnum
]
where
proxy :: Proxy a
proxy = Proxy

testFiniteLaws ::
forall a.
(NFData a, Show a, Eq a, Finite a, Typeable a) =>
Maybe (Dict (Bounded a)) ->
Maybe (Dict (Enum a)) ->
TestTree
testFiniteLaws mBounded mEnum =
testGroup (show (typeRep proxy)) $ finiteLaws mBounded mEnum
where
proxy :: Proxy a
proxy = Proxy

tests :: TestTree
tests = testGroup "Finite"
[ testFiniteLaws noBInst $ noEInst @Void
, testFiniteLaws hasBInst $ hasEInst @()
, testFiniteLaws hasBInst $ hasEInst @Bit
, testFiniteLaws hasBInst $ hasEInst @Bool
, testFiniteLaws hasBInst $ hasEInst @Ordering

, testFiniteLaws hasBInst $ hasEInst @Char
, testFiniteLaws hasBInst $ hasEInst @Int8
, testFiniteLaws hasBInst $ hasEInst @Int16
, testFiniteLaws hasBInst $ hasEInst @Word8
, testFiniteLaws hasBInst $ hasEInst @Word16

, testFiniteLaws hasBInst $ hasEInst @(BitVector 0)
, testFiniteLaws hasBInst $ hasEInst @(BitVector 1)
, testFiniteLaws hasBInst $ hasEInst @(BitVector 8)

, testFiniteLaws hasBInst $ hasEInst @(Index 0)
, testFiniteLaws hasBInst $ hasEInst @(Index 1)
, testFiniteLaws hasBInst $ hasEInst @(Index 128)

, testFiniteLaws hasBInst $ hasEInst @(Signed 0)
, testFiniteLaws hasBInst $ hasEInst @(Signed 1)
, testFiniteLaws hasBInst $ hasEInst @(Signed 8)

, testFiniteLaws hasBInst $ hasEInst @(Unsigned 0)
, testFiniteLaws hasBInst $ hasEInst @(Unsigned 1)
, testFiniteLaws hasBInst $ hasEInst @(Unsigned 8)

, testFiniteLaws noBInst $ noEInst @(Maybe (Index 0))
, testFiniteLaws noBInst $ noEInst @(Maybe (Index 1))
, testFiniteLaws noBInst $ noEInst @(Maybe (Index 27))

, testFiniteLaws noBInst $ noEInst @(Either Void (Index 0))
, testFiniteLaws noBInst $ noEInst @(Either Void (Index 1))
, testFiniteLaws noBInst $ noEInst @(Either Void (Index 27))
, testFiniteLaws noBInst $ noEInst @(Either Bool (Index 0))
, testFiniteLaws noBInst $ noEInst @(Either Bool (Index 1))
, testFiniteLaws noBInst $ noEInst @(Either Bool (Index 27))

, testFiniteLaws noBInst $ noEInst @(Compose Maybe Maybe Bool)
, testFiniteLaws hasBInst $ hasEInst @(Const Bool [Int])
#if MIN_VERSION_base(4,15,0)
, testFiniteLaws hasBInst $ hasEInst @(Down Bool)
#endif
, testFiniteLaws hasBInst $ hasEInst @(Identity Bool)
, testFiniteLaws noBInst $ noEInst @(Product Maybe Maybe Bit)
, testFiniteLaws noBInst $ noEInst @(Sum Maybe Maybe Bit)

, testFiniteLaws noBInst $ noEInst @(Vec 0 Void)
, testFiniteLaws noBInst $ noEInst @(Vec 1 Void)
, testFiniteLaws noBInst $ noEInst @(Vec 16 Void)
, testFiniteLaws noBInst $ noEInst @(Vec 0 Bool)
, testFiniteLaws noBInst $ noEInst @(Vec 1 Bool)
, testFiniteLaws noBInst $ noEInst @(Vec 16 Bool)

, testFiniteLaws noBInst $ noEInst @(RTree 0 Void)
, testFiniteLaws noBInst $ noEInst @(RTree 1 Void)
, testFiniteLaws noBInst $ noEInst @(RTree 4 Void)
, testFiniteLaws noBInst $ noEInst @(RTree 0 Bool)
, testFiniteLaws noBInst $ noEInst @(RTree 1 Bool)
, testFiniteLaws noBInst $ noEInst @(RTree 4 Bool)

, testFiniteLaws noBInst $ noEInst @(Void, Void)
, testFiniteLaws noBInst $ noEInst @(Bool, Void)
, testFiniteLaws noBInst $ noEInst @(Void, Bool)
, testFiniteLaws noBInst $ noEInst @(Bool, Bool)

, testFiniteLaws noBInst $ noEInst @(Bool, Bool, Bool)
, testFiniteLaws noBInst $ noEInst @(Void, Bool, Bool)
, testFiniteLaws noBInst $ noEInst @(Bool, Void, Bool)
, testFiniteLaws noBInst $ noEInst @(Bool, Bool, Void)

, testFiniteLaws noBInst $ noEInst @(Bool, Bool, Bool, Bool)
]
where
noBInst :: Maybe (Dict (Bounded a))
noBInst = Nothing

hasBInst :: Bounded a => Maybe (Dict (Bounded a))
hasBInst = Just Dict

noEInst :: Maybe (Dict (Enum a))
noEInst = Nothing

hasEInst :: Enum a => Maybe (Dict (Enum a))
hasEInst = Just Dict
2 changes: 2 additions & 0 deletions clash-prelude/tests/unittests.hs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ import qualified Clash.Tests.Vector
import qualified Clash.Tests.XException

import qualified Clash.Tests.Laws.Enum
import qualified Clash.Tests.Laws.Finite
import qualified Clash.Tests.Laws.SaturatingNum

tests :: TestTree
Expand Down Expand Up @@ -56,6 +57,7 @@ tests = testGroup "Unittests"
, Clash.Tests.XException.tests
, testGroup "Laws"
[ Clash.Tests.Laws.Enum.tests
, Clash.Tests.Laws.Finite.tests
, Clash.Tests.Laws.SaturatingNum.tests
]
]
Expand Down

0 comments on commit 645d9fa

Please sign in to comment.