{-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE RankNTypes #-} {-# LANGUAGE TypeApplications #-} module DenseBench ( benchSuite ) where import Prelude hiding (quotRem, gcd) import Gauge.Main import Data.Euclidean (Euclidean(..), GcdDomain(..), Field) import Data.Poly import qualified Data.Poly.Semiring as S (toPoly) import Data.Semiring (Semiring(..), Ring, Mod2(..)) import qualified Data.Semiring as S (fromIntegral) import qualified Data.Vector as V import qualified Data.Vector.Unboxed as U benchSuite :: Benchmark benchSuite = bgroup "dense" $ concat [ map benchAdd [100, 1000, 10000] , map benchMul [100, 1000, 10000] , map benchEval [100, 1000, 10000] , map benchDeriv [100, 1000, 10000] , map benchIntegral [100, 1000, 10000] , map benchQuotRem [10, 100] , map benchGcd [10, 100] , map benchGcdFracRat [10, 20, 40] , map benchGcdFracM [10, 100, 1000] ] benchAdd :: Int -> Benchmark benchAdd k = bench ("add/" ++ show k) $ nf (doBinOp (+)) k benchMul :: Int -> Benchmark benchMul k = bench ("mul/" ++ show k) $ nf (doBinOp (*)) k benchEval :: Int -> Benchmark benchEval k = bench ("eval/" ++ show k) $ nf doEval k benchDeriv :: Int -> Benchmark benchDeriv k = bench ("deriv/" ++ show k) $ nf doDeriv k benchIntegral :: Int -> Benchmark benchIntegral k = bench ("integral/" ++ show k) $ nf doIntegral k benchQuotRem :: Int -> Benchmark benchQuotRem k = bench ("quotRem/" ++ show k) $ nf doQuotRem k benchGcd :: Int -> Benchmark benchGcd k = bench ("gcd/" ++ show k) $ nf doGcd k benchGcdFracRat :: Int -> Benchmark benchGcdFracRat k = bench ("gcdFrac/Rational/" ++ show k) $ nf (doGcdFrac @Rational) k benchGcdFracM :: Int -> Benchmark benchGcdFracM k = bench ("gcdFrac/Mod2/" ++ show k) $ nf (getMod2 . doGcdFrac @Mod2) k doBinOp :: (forall a. Num a => a -> a -> a) -> Int -> Int doBinOp op n = U.sum zs where xs = toPoly $ U.generate n (* 2) ys = toPoly $ U.generate n (* 3) zs = unPoly $ xs `op` ys {-# INLINE doBinOp #-} doEval :: Int -> Int doEval n = eval xs n where xs = toPoly $ U.generate n (* 2) doDeriv :: Int -> Int doDeriv n = U.sum zs where xs = toPoly $ U.generate n (* 2) zs = unPoly $ deriv xs doIntegral :: Int -> Double doIntegral n = U.sum zs where xs = toPoly $ U.generate n ((* 2) . fromIntegral) zs = unPoly $ integral xs gen1 :: Ring a => Int -> a gen1 k = S.fromIntegral (truncate (pi * fromIntegral k :: Double) `mod` (k + 1)) gen2 :: Ring a => Int -> a gen2 k = S.fromIntegral (truncate (exp 1.0 * fromIntegral k :: Double) `mod` (k + 1)) doQuotRem :: Int -> Double doQuotRem n = U.sum (unPoly qs) + U.sum (unPoly rs) where xs = toPoly $ U.generate (2 * n) gen1 ys = toPoly $ U.generate n gen2 (qs, rs) = xs `quotRem` ys doGcd :: Int -> Integer doGcd n = V.sum gs where xs = toPoly $ V.generate n gen1 ys = toPoly $ V.generate n gen2 gs = unPoly $ xs `gcd` ys doGcdFrac :: (Eq a, Field a) => Int -> a doGcdFrac n = V.foldl' plus zero gs where xs = PolyOverField $ S.toPoly $ V.generate n gen1 ys = PolyOverField $ S.toPoly $ V.generate n gen2 gs = unPoly $ unPolyOverField $ xs `gcd` ys