{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE UndecidableInstances #-}
module Data.Vector.Fixed.Storable (
Vec
, Vec1
, Vec2
, Vec3
, Vec4
, Vec5
, unsafeFromForeignPtr
, unsafeToForeignPtr
, unsafeWith
, MVec(..)
, Storable
) where
import Control.Monad.Primitive
import Control.DeepSeq (NFData(..))
import Data.Monoid (Monoid(..))
import Data.Semigroup (Semigroup(..))
import Data.Data
import Foreign.Ptr (castPtr)
import Foreign.Storable
import Foreign.ForeignPtr
import Foreign.Marshal.Array ( copyArray, moveArray )
import GHC.ForeignPtr ( ForeignPtr(..), mallocPlainForeignPtrBytes )
import GHC.Ptr ( Ptr(..) )
import GHC.TypeLits
import Prelude ( Show(..),Eq(..),Ord(..),Num(..),Monad(..),IO,Int
, ($),undefined,seq)
import Data.Vector.Fixed hiding (index)
import Data.Vector.Fixed.Mutable
import qualified Data.Vector.Fixed.Cont as C
import qualified Data.Vector.Fixed.Internal as I
newtype Vec (n :: Nat) a = Vec (ForeignPtr a)
newtype MVec (n :: Nat) s a = MVec (ForeignPtr a)
deriving instance Typeable Vec
deriving instance Typeable MVec
type Vec1 = Vec 1
type Vec2 = Vec 2
type Vec3 = Vec 3
type Vec4 = Vec 4
type Vec5 = Vec 5
unsafeToForeignPtr :: Vec n a -> ForeignPtr a
{-# INLINE unsafeToForeignPtr #-}
unsafeToForeignPtr (Vec fp) = fp
unsafeFromForeignPtr :: ForeignPtr a -> Vec n a
{-# INLINE unsafeFromForeignPtr #-}
unsafeFromForeignPtr = Vec
unsafeWith :: (Ptr a -> IO b) -> Vec n a -> IO b
{-# INLINE unsafeWith #-}
unsafeWith f (Vec fp) = f (getPtr fp)
instance (Arity n, Storable a, Show a) => Show (Vec n a) where
showsPrec = I.showsPrec
instance (Arity n, Storable a, NFData a) => NFData (Vec n a) where
rnf = foldl (\r a -> r `seq` rnf a) ()
{-# INLINE rnf #-}
type instance Mutable (Vec n) = MVec n
instance (Arity n, Storable a) => MVector (MVec n) a where
new = unsafePrimToPrim $ do
fp <- mallocVector $ arity (Proxy :: Proxy n)
return $ MVec fp
{-# INLINE new #-}
copy (MVec fp) (MVec fq)
= unsafePrimToPrim
$ withForeignPtr fp $ \p ->
withForeignPtr fq $ \q ->
copyArray p q (arity (Proxy :: Proxy n))
{-# INLINE copy #-}
move (MVec fp) (MVec fq)
= unsafePrimToPrim
$ withForeignPtr fp $ \p ->
withForeignPtr fq $ \q ->
moveArray p q (arity (Proxy :: Proxy n))
{-# INLINE move #-}
unsafeRead (MVec fp) i
= unsafePrimToPrim
$ withForeignPtr fp (`peekElemOff` i)
{-# INLINE unsafeRead #-}
unsafeWrite (MVec fp) i x
= unsafePrimToPrim
$ withForeignPtr fp $ \p -> pokeElemOff p i x
{-# INLINE unsafeWrite #-}
instance (Arity n, Storable a) => IVector (Vec n) a where
unsafeFreeze (MVec fp) = return $ Vec fp
unsafeThaw (Vec fp) = return $ MVec fp
unsafeIndex (Vec fp) i
= unsafeInlineIO
$ withForeignPtr fp (`peekElemOff` i)
{-# INLINE unsafeFreeze #-}
{-# INLINE unsafeThaw #-}
{-# INLINE unsafeIndex #-}
type instance Dim (Vec n) = n
type instance DimM (MVec n) = n
instance (Arity n, Storable a) => Vector (Vec n) a where
construct = constructVec
inspect = inspectVec
basicIndex = index
{-# INLINE construct #-}
{-# INLINE inspect #-}
{-# INLINE basicIndex #-}
instance (Arity n, Storable a) => VectorN Vec n a
instance (Arity n, Storable a, Eq a) => Eq (Vec n a) where
(==) = eq
{-# INLINE (==) #-}
instance (Arity n, Storable a, Ord a) => Ord (Vec n a) where
compare = ord
{-# INLINE compare #-}
instance (Arity n, Storable a, Monoid a) => Monoid (Vec n a) where
mempty = replicate mempty
mappend = zipWith mappend
{-# INLINE mempty #-}
{-# INLINE mappend #-}
instance (Arity n, Storable a, Semigroup a) => Semigroup (Vec n a) where
(<>) = zipWith (<>)
{-# INLINE (<>) #-}
instance (Arity n, Storable a) => Storable (Vec n a) where
sizeOf _ = arity (Proxy :: Proxy n)
* sizeOf (undefined :: a)
alignment _ = alignment (undefined :: a)
peek ptr = do
arr@(MVec fp) <- new
withForeignPtr fp $ \p ->
moveArray p (castPtr ptr) (arity (Proxy :: Proxy n))
unsafeFreeze arr
poke ptr (Vec fp)
= withForeignPtr fp $ \p ->
moveArray (castPtr ptr) p (arity (Proxy :: Proxy n))
instance (Typeable n, Arity n, Storable a, Data a) => Data (Vec n a) where
gfoldl = C.gfoldl
gunfold = C.gunfold
toConstr _ = con_Vec
dataTypeOf _ = ty_Vec
ty_Vec :: DataType
ty_Vec = mkDataType "Data.Vector.Fixed.Primitive.Vec" [con_Vec]
con_Vec :: Constr
con_Vec = mkConstr ty_Vec "Vec" [] Prefix
mallocVector :: forall a. Storable a => Int -> IO (ForeignPtr a)
{-# INLINE mallocVector #-}
mallocVector size
= mallocPlainForeignPtrBytes (size * sizeOf (undefined :: a))
getPtr :: ForeignPtr a -> Ptr a
{-# INLINE getPtr #-}
getPtr (ForeignPtr addr _) = Ptr addr