{-# LANGUAGE NoImplicitPrelude,
MultiParamTypeClasses, FunctionalDependencies,
FlexibleInstances, FlexibleContexts, TypeFamilies,
UndecidableInstances, TypeSynonymInstances #-}
module Control.Monad.Memo.Vector.Unsafe
(
VectorCache,
VectorMemo,
unsafeEvalVectorMemo,
unsafeRunVectorMemo,
UVectorCache,
UVectorMemo,
unsafeEvalUVectorMemo,
unsafeRunUVectorMemo,
Container(..),
Cache,
genericUnsafeEvalVectorMemo,
genericUnsafeRunVectorMemo
) where
import Data.Function
import Data.Int
import Data.Maybe (Maybe(..))
import Data.Vector.Generic.Mutable
import qualified Data.Vector.Mutable as M
import qualified Data.Vector.Unboxed.Mutable as UM
import Control.Applicative
import Control.Monad
import Control.Monad.Fix
import Control.Monad.Trans.Class
import Control.Monad.Primitive
import Data.MaybeLike
import Control.Monad.Memo.Class
import Control.Monad.Trans.Memo.ReaderCache
newtype Container vec = Container { toVector :: vec }
type Cache vec k e = ReaderCache (Container (vec k e))
instance (PrimMonad m, PrimState m ~ s, MaybeLike e v, MVector c e) =>
MonadCache Int v (Cache c s e m) where
{-# INLINE lookup #-}
lookup k = do
c <- container
e <- lift $ unsafeRead (toVector c) k
return (if isNothing e then Nothing else Just (fromJust e))
{-# INLINE add #-}
add k v = do
c <- container
lift $ unsafeWrite (toVector c) k (just v)
instance (PrimMonad m, PrimState m ~ s, MaybeLike e v, MVector c e) =>
MonadMemo Int v (Cache c s e m) where
{-# INLINE memo #-}
memo f k = do
c <- container
e <- lift $ unsafeRead (toVector c) k
if isNothing e
then do
v <- f k
lift $ unsafeWrite (toVector c) k (just v)
return v
else return (fromJust e)
type Vector = M.MVector
type VectorCache s e = Cache Vector s e
class MaybeLike e v => VectorMemo v e | v -> e
unsafeEvalVectorMemo :: (PrimMonad m, VectorMemo v e) =>
VectorCache (PrimState m) e m a
-> Int
-> m a
{-# INLINE unsafeEvalVectorMemo #-}
unsafeEvalVectorMemo = genericUnsafeEvalVectorMemo
unsafeRunVectorMemo :: (PrimMonad m, VectorMemo v e) =>
VectorCache (PrimState m) e m a
-> Int
-> m (a, Vector (PrimState m) e)
{-# INLINE unsafeRunVectorMemo #-}
unsafeRunVectorMemo = genericUnsafeRunVectorMemo
type UVector = UM.MVector
type UVectorCache s e = Cache UVector s e
class MaybeLike e v => UVectorMemo v e | v -> e
unsafeEvalUVectorMemo :: (PrimMonad m, UVectorMemo v e, MVector UVector e) =>
UVectorCache (PrimState m) e m a
-> Int
-> m a
{-# INLINE unsafeEvalUVectorMemo #-}
unsafeEvalUVectorMemo = genericUnsafeEvalVectorMemo
unsafeRunUVectorMemo :: (PrimMonad m, UVectorMemo v e, MVector UVector e) =>
UVectorCache (PrimState m) e m a
-> Int
-> m (a, UVector (PrimState m) e)
{-# INLINE unsafeRunUVectorMemo #-}
unsafeRunUVectorMemo = genericUnsafeRunVectorMemo
genericUnsafeEvalVectorMemo :: (MaybeLike e v, PrimMonad m, MVector c e) =>
Cache c (PrimState m) e m a -> Int -> m a
{-# INLINE genericUnsafeEvalVectorMemo #-}
genericUnsafeEvalVectorMemo m n = do
vec <- replicate n nothing
evalReaderCache m (Container vec)
genericUnsafeRunVectorMemo :: (MaybeLike e v, PrimMonad m, MVector c e) =>
Cache c (PrimState m) e m a -> Int -> m (a, c (PrimState m) e)
{-# INLINE genericUnsafeRunVectorMemo #-}
genericUnsafeRunVectorMemo m n = do
vec <- replicate n nothing
a <- evalReaderCache m (Container vec)
return (a, vec)