{-# OPTIONS_GHC -Wno-redundant-constraints #-}
{-# LANGUAGE FlexibleInstances, FlexibleContexts #-}
{-# LANGUAGE DefaultSignatures, TypeOperators #-}
module Language.Souffle.Marshal
( Marshal(..)
, MonadPush(..)
, MonadPop(..)
) where
import GHC.Generics
import Data.Int
import Data.Word
import qualified Data.Text as T
import qualified Data.Text.Lazy as TL
import qualified Language.Souffle.Internal.Constraints as C
class Monad m => MonadPush m where
pushInt32 :: Int32 -> m ()
pushUInt32 :: Word32 -> m ()
pushFloat :: Float -> m ()
pushString :: String -> m ()
class Monad m => MonadPop m where
popInt32 :: m Int32
popUInt32 :: m Word32
popFloat :: m Float
popString :: m String
class Marshal a where
push :: MonadPush m => a -> m ()
pop :: MonadPop m => m a
default push
:: (Generic a, C.SimpleProduct a, GMarshal (Rep a), MonadPush m)
=> a -> m ()
default pop
:: (Generic a, C.SimpleProduct a, GMarshal (Rep a), MonadPop m)
=> m a
push a = gpush (from a)
{-# INLINABLE push #-}
pop = to <$> gpop
{-# INLINABLE pop #-}
instance Marshal Int32 where
push = pushInt32
{-# INLINABLE push #-}
pop = popInt32
{-# INLINABLE pop #-}
instance Marshal Word32 where
push = pushUInt32
{-# INLINABLE push #-}
pop = popUInt32
{-# INLINABLE pop #-}
instance Marshal Float where
push = pushFloat
{-# INLINABLE push #-}
pop = popFloat
{-# INLINABLE pop #-}
instance Marshal String where
push = pushString
{-# INLINABLE push #-}
pop = popString
{-# INLINABLE pop #-}
instance Marshal T.Text where
push = push . T.unpack
{-# INLINABLE push #-}
pop = T.pack <$> pop
{-# INLINABLE pop #-}
instance Marshal TL.Text where
push = push . TL.unpack
{-# INLINABLE push #-}
pop = TL.pack <$> pop
{-# INLINABLE pop #-}
class GMarshal f where
gpush :: MonadPush m => f a -> m ()
gpop :: MonadPop m => m (f a)
instance Marshal a => GMarshal (K1 i a) where
gpush (K1 x) = push x
{-# INLINABLE gpush #-}
gpop = K1 <$> pop
{-# INLINABLE gpop #-}
instance (GMarshal f, GMarshal g) => GMarshal (f :*: g) where
gpush (a :*: b) = do
gpush a
gpush b
{-# INLINABLE gpush #-}
gpop = (:*:) <$> gpop <*> gpop
{-# INLINABLE gpop #-}
instance GMarshal a => GMarshal (M1 i c a) where
gpush (M1 x) = gpush x
{-# INLINABLE gpush #-}
gpop = M1 <$> gpop
{-# INLINABLE gpop #-}