{-# LANGUAGE CPP #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE UndecidableInstances #-}
{-# OPTIONS_GHC -fplugin GHC.TypeLits.KnownNat.Solver #-}
module Clash.Class.Exp (Exp, ExpResult, (^)) where
import qualified Prelude as P
import Prelude hiding ((^))
import Clash.Annotations.Primitive (hasBlackBox)
import Clash.Promoted.Nat (SNat(..), snatToInteger)
import Clash.Sized.Internal.Index (Index)
import Clash.Sized.Internal.Signed (Signed)
import Clash.Sized.Internal.Unsigned (Unsigned)
import GHC.TypeLits
(KnownNat, Nat, type (^), type (*))
import GHC.TypeLits.Extra (Max)
class Exp a where
type ExpResult a (n :: Nat)
(^)
:: a
-> SNat n
-> ExpResult a n
instance KnownNat m => Exp (Index m) where
type ExpResult (Index m) n = Index (Max 2 (m ^ n))
^ :: Index m -> SNat n -> ExpResult (Index m) n
(^) = Index m -> SNat n -> ExpResult (Index m) n
forall (m :: Nat) (n :: Nat).
KnownNat m =>
Index m -> SNat n -> Index (Max 2 (m ^ n))
expIndex#
{-# INLINE (^) #-}
instance KnownNat m => Exp (Signed m) where
type ExpResult (Signed m) n = Signed (Max 2 (m * n))
^ :: Signed m -> SNat n -> ExpResult (Signed m) n
(^) = Signed m -> SNat n -> ExpResult (Signed m) n
forall (m :: Nat) (n :: Nat).
KnownNat m =>
Signed m -> SNat n -> Signed (Max 2 (m * n))
expSigned#
{-# INLINE (^) #-}
instance KnownNat m => Exp (Unsigned m) where
type ExpResult (Unsigned m) n = Unsigned (Max 1 (m * n))
^ :: Unsigned m -> SNat n -> ExpResult (Unsigned m) n
(^) = Unsigned m -> SNat n -> ExpResult (Unsigned m) n
forall (m :: Nat) (n :: Nat).
KnownNat m =>
Unsigned m -> SNat n -> Unsigned (Max 1 (m * n))
expUnsigned#
{-# INLINE (^) #-}
expIndex#
:: KnownNat m
=> Index m
-> SNat n
-> Index (Max 2 (m ^ n))
expIndex# :: Index m -> SNat n -> Index (Max 2 (m ^ n))
expIndex# Index m
b e :: SNat n
e@SNat n
SNat =
Integer -> Index (Max 2 (m ^ n))
forall a. Num a => Integer -> a
fromInteger (Index m -> Integer
forall a. Integral a => a -> Integer
toInteger Index m
b Integer -> Integer -> Integer
forall a b. (Num a, Integral b) => a -> b -> a
P.^ SNat n -> Integer
forall (n :: Nat). SNat n -> Integer
snatToInteger SNat n
e)
{-# NOINLINE expIndex# #-}
{-# ANN expIndex# hasBlackBox #-}
expSigned#
:: KnownNat m
=> Signed m
-> SNat n
-> Signed (Max 2 (m * n))
expSigned# :: Signed m -> SNat n -> Signed (Max 2 (m * n))
expSigned# Signed m
b e :: SNat n
e@SNat n
SNat =
Integer -> Signed (Max 2 (m * n))
forall a. Num a => Integer -> a
fromInteger (Signed m -> Integer
forall a. Integral a => a -> Integer
toInteger Signed m
b Integer -> Integer -> Integer
forall a b. (Num a, Integral b) => a -> b -> a
P.^ SNat n -> Integer
forall (n :: Nat). SNat n -> Integer
snatToInteger SNat n
e)
{-# NOINLINE expSigned# #-}
{-# ANN expSigned# hasBlackBox #-}
expUnsigned#
:: KnownNat m
=> Unsigned m
-> SNat n
-> Unsigned (Max 1 (m * n))
expUnsigned# :: Unsigned m -> SNat n -> Unsigned (Max 1 (m * n))
expUnsigned# Unsigned m
b e :: SNat n
e@SNat n
SNat =
Integer -> Unsigned (Max 1 (m * n))
forall a. Num a => Integer -> a
fromInteger (Unsigned m -> Integer
forall a. Integral a => a -> Integer
toInteger Unsigned m
b Integer -> Integer -> Integer
forall a b. (Num a, Integral b) => a -> b -> a
P.^ SNat n -> Integer
forall (n :: Nat). SNat n -> Integer
snatToInteger SNat n
e)
{-# NOINLINE expUnsigned# #-}
{-# ANN expUnsigned# hasBlackBox #-}