{-# LANGUAGE DataKinds #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE Trustworthy #-}
{-# OPTIONS_GHC -fplugin GHC.TypeLits.Normalise #-}
{-# OPTIONS_HADDOCK show-extensions #-}
module CLaSH.Promoted.Nat
( SNat (..), snat, withSNat, snatToInteger, addSNat, subSNat, mulSNat, powSNat
, UNat (..), toUNat, addUNat, multUNat, powUNat
)
where
import Data.Proxy
import GHC.TypeLits
import Unsafe.Coerce
data SNat (n :: Nat) = KnownNat n => SNat (Proxy n)
instance Show (SNat n) where
show (SNat p) = 'd' : show (natVal p)
{-# INLINE snat #-}
snat :: KnownNat n => SNat n
snat = SNat Proxy
{-# INLINE withSNat #-}
withSNat :: KnownNat n => (SNat n -> a) -> a
withSNat f = f (SNat Proxy)
{-# INLINE snatToInteger #-}
snatToInteger :: SNat n -> Integer
snatToInteger (SNat p) = natVal p
data UNat :: Nat -> * where
UZero :: UNat 0
USucc :: UNat n -> UNat (n + 1)
toUNat :: SNat n -> UNat n
toUNat (SNat p) = fromI (natVal p)
where
fromI :: Integer -> UNat m
fromI 0 = unsafeCoerce UZero
fromI n = unsafeCoerce (USucc (fromI (n - 1)))
addUNat :: UNat n -> UNat m -> UNat (n + m)
addUNat UZero y = y
addUNat x UZero = x
addUNat (USucc x) y = USucc (addUNat x y)
multUNat :: UNat n -> UNat m -> UNat (n * m)
multUNat UZero _ = UZero
multUNat _ UZero = UZero
multUNat (USucc x) y = addUNat y (multUNat x y)
powUNat :: UNat n -> UNat m -> UNat (n ^ m)
powUNat _ UZero = USucc UZero
powUNat x (USucc y) = multUNat x (powUNat x y)
addSNat :: KnownNat (a + b) => SNat a -> SNat b -> SNat (a+b)
addSNat _ _ = snat
subSNat :: KnownNat (a - b) => SNat a -> SNat b -> SNat (a-b)
subSNat _ _ = snat
mulSNat :: KnownNat (a * b) => SNat a -> SNat b -> SNat (a*b)
mulSNat _ _ = snat
powSNat :: KnownNat (a ^ b) => SNat a -> SNat b -> SNat (a^b)
powSNat _ _ = snat