module Numeric.NumType.DK.Naturals where
import Prelude hiding (pred)
infixr 8 ^
infixl 7 *
infixl 6 +
data Nat = Z | S Nat
type family (n::Nat) + (n'::Nat) :: Nat where
n + 'Z = n
n + 'S n' = 'S n + n'
type family (n::Nat) (n'::Nat) :: Nat where
n 'Z = n
'S n 'S n' = n n'
type family (n::Nat) * (n'::Nat) :: Nat
where
n * 'Z = 'Z
n * ('S n') = n + n * n'
type family (n::Nat) ^ (n'::Nat) :: Nat
where
n ^ 'Z = 'S 'Z
n ^ 'S n' = n * n ^ n'
class KnownNat (n::Nat) where natVal :: proxy n -> Integer
instance KnownNat 'Z where natVal _ = 0
instance KnownNat n => KnownNat ('S n) where
natVal = (1 +) . natVal . pred
where
pred :: proxy ('S n) -> proxy n
pred = undefined