{- | Here we show an example of how to define a Storable instance with this module. > import Foreign.Storable.Record as Store > import Foreign.Storable (Storable (..), ) > > import Control.Applicative (liftA2, ) > > data Stereo a = Stereo {left, right :: a} > > store :: Storable a => Store.Dictionary (Stereo a) > store = > Store.run $ > liftA2 Stereo > (Store.element left) > (Store.element right) > > instance (Storable a) => Storable (Stereo a) where > sizeOf = Store.sizeOf store > alignment = Store.alignment store > peek = Store.peek store > poke = Store.poke store The @Stereo@ constructor is exclusively used for constructing the @peek@ function, whereas the accessors in the @element@ calls are used for assembling the @poke@ function. It is required that the order of arguments of @Stereo@ matches the record accessors in the @element@ calls. If you want that the stored data correctly and fully represents your Haskell data, it must hold: > Stereo (left x) (right x) = x . Unfortunately this cannot be checked automatically. However, mismatching types that are caused by swapped arguments are detected by the type system. Our system performs for you: Size and alignment computation, poking and peeking. Thus several inconsistency bugs can be prevented using this package, like size mismatching the space required by @poke@ actions. There is no more restriction, thus smart constructors and accessors and nested records work, too. For nested records however, I recommend individual Storable instances for the sub-records. You see it would simplify class instantiation if we could tell the class dictionary at once instead of defining each method separately. In this implementation we tail pad records according to the overall required alignment in conformance to the Linux/X86 ABI. -} module Foreign.Storable.RecordReaderPtr ( Dictionary, Access, element, run, alignment, sizeOf, peek, poke, ) where import Control.Monad.Trans.Reader (ReaderT(ReaderT), runReaderT, Reader, reader, runReader, ) import Control.Monad.Trans.Writer (Writer, writer, runWriter, ) import Control.Monad.Trans.State (State, modify, get, runState, ) import Control.Applicative (Applicative(pure, (<*>)), ) import Data.Functor.Compose (Compose(Compose), ) import Data.Monoid (Monoid(mempty, mappend), ) import Data.Semigroup (Semigroup((<>)), ) import Foreign.Storable.FixedArray (roundUp, ) import qualified Foreign.Storable as St import Foreign.Ptr (Ptr, ) import Foreign.Storable (Storable, ) data Dictionary r = Dictionary { sizeOf_ :: Int, alignment_ :: Alignment, ptrBox :: Reader (Ptr r) (Box r r) } newtype Access r a = Access (Compose (Writer Alignment) (Compose (State Int) (Compose (Reader (Ptr r)) (Box r))) a) instance Functor (Access r) where {-# INLINE fmap #-} fmap f (Access m) = Access (fmap f m) instance Applicative (Access r) where {-# INLINE pure #-} {-# INLINE (<*>) #-} pure a = Access (pure a) Access f <*> Access x = Access (f <*> x) data Box r a = Box { peek_ :: IO a, poke_ :: ReaderT r IO () } instance Functor (Box r) where {-# INLINE fmap #-} fmap f (Box pe po) = Box (fmap f pe) po instance Applicative (Box r) where {-# INLINE pure #-} {-# INLINE (<*>) #-} pure a = Box (pure a) (pure ()) f <*> x = Box (peek_ f <*> peek_ x) (poke_ f >> poke_ x) newtype Alignment = Alignment {deconsAlignment :: Int} instance Semigroup Alignment where {-# INLINE (<>) #-} Alignment x <> Alignment y = Alignment (lcm x y) instance Monoid Alignment where {-# INLINE mempty #-} {-# INLINE mappend #-} mempty = Alignment 1 mappend = (<>) {-# INLINE element #-} element :: Storable a => (r -> a) -> Access r a element f = let align = St.alignment (f (error "Storable.Record.element.alignment: content touched")) size = St.sizeOf (f (error "Storable.Record.element.size: content touched")) in Access $ Compose $ writer $ flip (,) (Alignment align) $ Compose $ do modify (roundUp align) offset <- get modify (+size) return $ Compose $ reader $ \ptr -> Box (St.peekByteOff ptr offset) (ReaderT $ St.pokeByteOff ptr offset . f) {-# INLINE run #-} run :: Access r r -> Dictionary r run (Access (Compose m)) = let (Compose s, align) = runWriter m (Compose r, size) = runState s 0 in Dictionary (roundUp (deconsAlignment align) size) align r {-# INLINE alignment #-} alignment :: Dictionary r -> r -> Int alignment dict _ = deconsAlignment $ alignment_ dict {-# INLINE sizeOf #-} sizeOf :: Dictionary r -> r -> Int sizeOf dict _ = sizeOf_ dict {-# INLINE peek #-} peek :: Dictionary r -> Ptr r -> IO r peek dict ptr = peek_ $ runReader (ptrBox dict) ptr {-# INLINE poke #-} poke :: Dictionary r -> Ptr r -> r -> IO () poke dict ptr = runReaderT (poke_ $ runReader (ptrBox dict) ptr)