-- | -- Module : Basement.UArray.Mutable -- License : BSD-style -- Maintainer : Vincent Hanquez -- Stability : experimental -- Portability : portable -- -- A simple array abstraction that allow to use typed -- array of bytes where the array is pinned in memory -- to allow easy use with Foreign interfaces, ByteString -- and always aligned to 64 bytes. -- {-# LANGUAGE MagicHash #-} {-# LANGUAGE UnboxedTuples #-} {-# LANGUAGE ScopedTypeVariables #-} module Basement.UArray.Mutable ( MUArray(..) -- * Property queries , sizeInMutableBytesOfContent , mutableLength , mutableOffset , mutableSame , onMutableBackend -- * Allocation & Copy , new , newPinned , newNative , mutableForeignMem , copyAt , copyFromPtr , copyToPtr , sub -- , copyAddr -- * Reading and Writing cells , unsafeWrite , unsafeRead , write , read , withMutablePtr ) where import GHC.Prim import GHC.Types import GHC.Ptr import Basement.Compat.Base import Basement.Compat.Primitive import Data.Proxy import Basement.Types.OffsetSize import Basement.Monad import Basement.PrimType import Basement.FinalPtr import Basement.Exception import qualified Basement.Block.Mutable as MBLK import Basement.Block (MutableBlock(..)) import Basement.UArray.Base hiding (empty) import Basement.Numerical.Subtractive import Foreign.Marshal.Utils (copyBytes) sizeInMutableBytesOfContent :: forall ty s . PrimType ty => MUArray ty s -> CountOf Word8 sizeInMutableBytesOfContent _ = primSizeInBytes (Proxy :: Proxy ty) {-# INLINE sizeInMutableBytesOfContent #-} -- | read a cell in a mutable array. -- -- If the index is out of bounds, an error is raised. read :: (PrimMonad prim, PrimType ty) => MUArray ty (PrimState prim) -> Offset ty -> prim ty read array n | isOutOfBound n len = primOutOfBound OOB_Read n len | otherwise = unsafeRead array n where len = mutableLength array {-# INLINE read #-} -- | Write to a cell in a mutable array. -- -- If the index is out of bounds, an error is raised. write :: (PrimMonad prim, PrimType ty) => MUArray ty (PrimState prim) -> Offset ty -> ty -> prim () write array n val | isOutOfBound n len = primOutOfBound OOB_Write n len | otherwise = unsafeWrite array n val where len = mutableLength array {-# INLINE write #-} empty :: (PrimType ty, PrimMonad prim) => prim (MUArray ty (PrimState prim)) empty = MUArray 0 0 . MUArrayMBA <$> MBLK.mutableEmpty mutableSame :: MUArray ty st -> MUArray ty st -> Bool mutableSame (MUArray sa ea (MUArrayMBA (MutableBlock ma))) (MUArray sb eb (MUArrayMBA (MutableBlock mb))) = (sa == sb) && (ea == eb) && bool# (sameMutableByteArray# ma mb) mutableSame (MUArray s1 e1 (MUArrayAddr f1)) (MUArray s2 e2 (MUArrayAddr f2)) = (s1 == s2) && (e1 == e2) && finalPtrSameMemory f1 f2 mutableSame _ _ = False mutableForeignMem :: (PrimMonad prim, PrimType ty) => FinalPtr ty -- ^ the start pointer with a finalizer -> Int -- ^ the number of elements (in elements, not bytes) -> prim (MUArray ty (PrimState prim)) mutableForeignMem fptr nb = pure $ MUArray (Offset 0) (CountOf nb) (MUArrayAddr fptr) sub :: (PrimMonad prim, PrimType ty) => MUArray ty (PrimState prim) -> Int -- The number of elements to drop ahead -> Int -- Then the number of element to retain -> prim (MUArray ty (PrimState prim)) sub (MUArray start sz back) dropElems' takeElems | takeElems <= 0 = empty | Just keepElems <- sz - dropElems, keepElems > 0 = pure $ MUArray (start `offsetPlusE` dropElems) (min (CountOf takeElems) keepElems) back | otherwise = empty where dropElems = max 0 (CountOf dropElems') -- | return the numbers of elements in a mutable array mutableLength :: PrimType ty => MUArray ty st -> CountOf ty mutableLength (MUArray _ end _) = end withMutablePtrHint :: forall ty prim a . (PrimMonad prim, PrimType ty) => Bool -> Bool -> MUArray ty (PrimState prim) -> (Ptr ty -> prim a) -> prim a withMutablePtrHint _ _ (MUArray start _ (MUArrayAddr fptr)) f = withFinalPtr fptr (\ptr -> f (ptr `plusPtr` os)) where sz = primSizeInBytes (Proxy :: Proxy ty) !(Offset os) = offsetOfE sz start withMutablePtrHint skipCopy skipCopyBack vec@(MUArray start vecSz (MUArrayMBA (MutableBlock a))) f | isMutablePinned vec == Pinned = mutableByteArrayContent a >>= \ptr -> f (ptr `plusPtr` os) | otherwise = do trampoline <- newPinned vecSz if not skipCopy then copyAt trampoline 0 vec 0 vecSz else pure () r <- withMutablePtr trampoline f if not skipCopyBack then copyAt vec 0 trampoline 0 vecSz else pure () pure r where !(Offset os) = offsetOfE sz start sz = primSizeInBytes (Proxy :: Proxy ty) mutableByteArrayContent :: PrimMonad prim => MutableByteArray# (PrimState prim) -> prim (Ptr ty) mutableByteArrayContent mba = primitive $ \s1 -> case unsafeFreezeByteArray# mba s1 of (# s2, ba #) -> (# s2, Ptr (byteArrayContents# ba) #) -- | Create a pointer on the beginning of the mutable array -- and call a function 'f'. -- -- The mutable buffer can be mutated by the 'f' function -- and the change will be reflected in the mutable array -- -- If the mutable array is unpinned, a trampoline buffer -- is created and the data is only copied when 'f' return. withMutablePtr :: (PrimMonad prim, PrimType ty) => MUArray ty (PrimState prim) -> (Ptr ty -> prim a) -> prim a withMutablePtr = withMutablePtrHint False False -- | Copy from a pointer, @count@ elements, into the mutable array copyFromPtr :: forall prim ty . (PrimMonad prim, PrimType ty) => Ptr ty -> CountOf ty -> MUArray ty (PrimState prim) -> prim () copyFromPtr src@(Ptr src#) count marr | count > arrSz = primOutOfBound OOB_MemCopy (sizeAsOffset count) arrSz | otherwise = onMutableBackend copyNative copyPtr marr where arrSz = mutableLength marr ofs = mutableOffset marr sz = primSizeInBytes (Proxy :: Proxy ty) !(CountOf bytes@(I# bytes#)) = sizeOfE sz count !(Offset od@(I# od#)) = offsetOfE sz ofs copyNative mba = primitive $ \st -> (# copyAddrToByteArray# src# mba od# bytes# st, () #) copyPtr fptr = withFinalPtr fptr $ \dst -> unsafePrimFromIO $ copyBytes (dst `plusPtr` od) src bytes -- | Copy all the block content to the memory starting at the destination address copyToPtr :: forall ty prim . (PrimType ty, PrimMonad prim) => MUArray ty (PrimState prim) -- ^ the source mutable array to copy -> Ptr ty -- ^ The destination address where the copy is going to start -> prim () copyToPtr marr dst@(Ptr dst#) = onMutableBackend copyNative copyPtr marr where copyNative mba = primitive $ \s1 -> case unsafeFreezeByteArray# mba s1 of (# s2, ba #) -> (# compatCopyByteArrayToAddr# ba os# dst# szBytes# s2, () #) copyPtr fptr = unsafePrimFromIO $ withFinalPtr fptr $ \ptr -> copyBytes dst (ptr `plusPtr` os) szBytes !(Offset os@(I# os#)) = offsetInBytes $ mutableOffset marr !(CountOf szBytes@(I# szBytes#)) = sizeInBytes $ mutableLength marr mutableOffset :: MUArray ty st -> Offset ty mutableOffset (MUArray ofs _ _) = ofs