{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE EmptyDataDecls #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE PolyKinds #-}
{-# LANGUAGE Rank2Types #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeSynonymInstances #-}
{-# OPTIONS_HADDOCK not-home #-}
module Numeric.Rounded.Rounding
( Rounding(..)
, RoundingMode(..)
, reifyRounding
) where
import Data.Data
import Data.Singletons
import Numeric.MPFR.Types
data RoundingMode
= TowardNearestWithTiesAwayFromZero
| TowardNearest
| TowardZero
| TowardInf
| TowardNegInf
| AwayFromZero
| Faithfully
deriving (Eq,Ord,Show,Read,Data,Typeable)
class Rounding (r :: RoundingMode) where rounding :: Proxy r -> RoundingMode
instance Rounding TowardNearest where rounding _ = TowardNearest
instance Rounding TowardZero where rounding _ = TowardZero
instance Rounding TowardInf where rounding _ = TowardInf
instance Rounding TowardNegInf where rounding _ = TowardNegInf
instance Rounding AwayFromZero where rounding _ = AwayFromZero
instance Rounding Faithfully where rounding _ = Faithfully
instance Rounding TowardNearestWithTiesAwayFromZero where rounding _ = TowardNearestWithTiesAwayFromZero
instance Enum RoundingMode where
toEnum MPFR_RNDNA = TowardNearestWithTiesAwayFromZero
toEnum MPFR_RNDN = TowardNearest
toEnum MPFR_RNDZ = TowardZero
toEnum MPFR_RNDU = TowardInf
toEnum MPFR_RNDD = TowardNegInf
toEnum MPFR_RNDA = AwayFromZero
toEnum MPFR_RNDF = Faithfully
toEnum _ = error "out of range"
fromEnum TowardNearestWithTiesAwayFromZero = MPFR_RNDNA
fromEnum TowardNearest = MPFR_RNDN
fromEnum TowardZero = MPFR_RNDZ
fromEnum TowardInf = MPFR_RNDU
fromEnum TowardNegInf = MPFR_RNDD
fromEnum AwayFromZero = MPFR_RNDA
fromEnum Faithfully = MPFR_RNDF
instance Bounded RoundingMode where
minBound = TowardNearestWithTiesAwayFromZero
maxBound = Faithfully
newtype instance Sing (m :: RoundingMode) = SRounding RoundingMode
instance SingI TowardNearestWithTiesAwayFromZero where sing = SRounding TowardNearestWithTiesAwayFromZero
instance SingI TowardNearest where sing = SRounding TowardNearest
instance SingI TowardZero where sing = SRounding TowardZero
instance SingI TowardInf where sing = SRounding TowardInf
instance SingI TowardNegInf where sing = SRounding TowardNegInf
instance SingI AwayFromZero where sing = SRounding AwayFromZero
instance SingI Faithfully where sing = SRounding Faithfully
reifyRounding :: RoundingMode -> (forall s. Rounding s => Proxy s -> r) -> r
reifyRounding TowardNearestWithTiesAwayFromZero f = f (Proxy :: Proxy TowardNearestWithTiesAwayFromZero)
reifyRounding TowardNearest f = f (Proxy :: Proxy TowardNearest)
reifyRounding TowardZero f = f (Proxy :: Proxy TowardZero)
reifyRounding TowardInf f = f (Proxy :: Proxy TowardInf)
reifyRounding TowardNegInf f = f (Proxy :: Proxy TowardNegInf)
reifyRounding AwayFromZero f = f (Proxy :: Proxy AwayFromZero)
reifyRounding Faithfully f = f (Proxy :: Proxy Faithfully)
{-# INLINE reifyRounding #-}