{-# LANGUAGE CPP #-}
{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE UnboxedTuples #-}
{-# LANGUAGE Trustworthy #-}
{-# LANGUAGE ScopedTypeVariables #-}
module Utils
( roundTo
, i2d
, maxExpt
, magnitude
) where
import GHC.Base (Int(I#), Char(C#), chr#, ord#, (+#))
import qualified Data.Primitive.Array as Primitive
import Control.Monad.ST (runST)
#if MIN_VERSION_base(4,5,0)
import Data.Bits (unsafeShiftR)
#else
import Data.Bits (shiftR)
#endif
roundTo :: Int -> [Int] -> (Int, [Int])
roundTo :: Int -> [Int] -> (Int, [Int])
roundTo Int
d [Int]
is =
case Int -> Bool -> [Int] -> (Int, [Int])
f Int
d Bool
True [Int]
is of
x :: (Int, [Int])
x@(Int
0,[Int]
_) -> (Int, [Int])
x
(Int
1,[Int]
xs) -> (Int
1, Int
1forall a. a -> [a] -> [a]
:[Int]
xs)
(Int, [Int])
_ -> forall a. HasCallStack => [Char] -> a
error [Char]
"roundTo: bad Value"
where
base :: Int
base = Int
10
b2 :: Int
b2 = Int
base forall a. Integral a => a -> a -> a
`quot` Int
2
f :: Int -> Bool -> [Int] -> (Int, [Int])
f Int
n Bool
_ [] = (Int
0, forall a. Int -> a -> [a]
replicate Int
n Int
0)
f Int
0 Bool
e (Int
x:[Int]
xs) | Int
x forall a. Eq a => a -> a -> Bool
== Int
b2 Bool -> Bool -> Bool
&& Bool
e Bool -> Bool -> Bool
&& forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all (forall a. Eq a => a -> a -> Bool
== Int
0) [Int]
xs = (Int
0, [])
| Bool
otherwise = (if Int
x forall a. Ord a => a -> a -> Bool
>= Int
b2 then Int
1 else Int
0, [])
f Int
n Bool
_ (Int
i:[Int]
xs)
| Int
i' forall a. Eq a => a -> a -> Bool
== Int
base = (Int
1,Int
0forall a. a -> [a] -> [a]
:[Int]
ds)
| Bool
otherwise = (Int
0,Int
i'forall a. a -> [a] -> [a]
:[Int]
ds)
where
(Int
c,[Int]
ds) = Int -> Bool -> [Int] -> (Int, [Int])
f (Int
nforall a. Num a => a -> a -> a
-Int
1) (forall a. Integral a => a -> Bool
even Int
i) [Int]
xs
i' :: Int
i' = Int
c forall a. Num a => a -> a -> a
+ Int
i
{-# INLINE i2d #-}
i2d :: Int -> Char
i2d :: Int -> Char
i2d (I# Int#
i#) = Char# -> Char
C# (Int# -> Char#
chr# (Char# -> Int#
ord# Char#
'0'# Int# -> Int# -> Int#
+# Int#
i# ))
maxExpt :: Int
maxExpt :: Int
maxExpt = Int
324
expts10 :: Primitive.Array Integer
expts10 :: Array Integer
expts10 = forall a. (forall s. ST s a) -> a
runST forall a b. (a -> b) -> a -> b
$ do
MutableArray (PrimState (ST s)) Integer
ma <- forall (m :: * -> *) a.
PrimMonad m =>
Int -> a -> m (MutableArray (PrimState m) a)
Primitive.newArray Int
maxExpt forall error. error
uninitialised
forall (m :: * -> *) a.
PrimMonad m =>
MutableArray (PrimState m) a -> Int -> a -> m ()
Primitive.writeArray MutableArray (PrimState (ST s)) Integer
ma Int
0 Integer
1
forall (m :: * -> *) a.
PrimMonad m =>
MutableArray (PrimState m) a -> Int -> a -> m ()
Primitive.writeArray MutableArray (PrimState (ST s)) Integer
ma Int
1 Integer
10
let go :: Int -> ST s (Array Integer)
go !Int
ix
| Int
ix forall a. Eq a => a -> a -> Bool
== Int
maxExpt = forall (m :: * -> *) a.
PrimMonad m =>
MutableArray (PrimState m) a -> m (Array a)
Primitive.unsafeFreezeArray MutableArray (PrimState (ST s)) Integer
ma
| Bool
otherwise = do
forall (m :: * -> *) a.
PrimMonad m =>
MutableArray (PrimState m) a -> Int -> a -> m ()
Primitive.writeArray MutableArray (PrimState (ST s)) Integer
ma Int
ix Integer
xx
forall (m :: * -> *) a.
PrimMonad m =>
MutableArray (PrimState m) a -> Int -> a -> m ()
Primitive.writeArray MutableArray (PrimState (ST s)) Integer
ma (Int
ixforall a. Num a => a -> a -> a
+Int
1) (Integer
10forall a. Num a => a -> a -> a
*Integer
xx)
Int -> ST s (Array Integer)
go (Int
ixforall a. Num a => a -> a -> a
+Int
2)
where
xx :: Integer
xx = Integer
x forall a. Num a => a -> a -> a
* Integer
x
x :: Integer
x = forall a. Array a -> Int -> a
Primitive.indexArray Array Integer
expts10 Int
half
#if MIN_VERSION_base(4,5,0)
!half :: Int
half = Int
ix forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
1
#else
!half = ix `shiftR` 1
#endif
Int -> ST s (Array Integer)
go Int
2
uninitialised :: error
uninitialised :: forall error. error
uninitialised = forall a. HasCallStack => [Char] -> a
error [Char]
"Data.Scientific: uninitialised element"
magnitude :: Num a => Int -> a
magnitude :: forall a. Num a => Int -> a
magnitude Int
e | Int
e forall a. Ord a => a -> a -> Bool
< Int
maxExpt = Int -> a
cachedPow10 Int
e
| Bool
otherwise = Int -> a
cachedPow10 Int
hi forall a. Num a => a -> a -> a
* a
10 forall a b. (Num a, Integral b) => a -> b -> a
^ (Int
e forall a. Num a => a -> a -> a
- Int
hi)
where
cachedPow10 :: Int -> a
cachedPow10 = forall a. Num a => Integer -> a
fromInteger forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. Array a -> Int -> a
Primitive.indexArray Array Integer
expts10
hi :: Int
hi = Int
maxExpt forall a. Num a => a -> a -> a
- Int
1