| Copyright | (c) 2023 Composewell Technologies |
|---|---|
| License | BSD3-3-Clause |
| Maintainer | streamly@composewell.com |
| Portability | GHC |
| Safe Haskell | None |
| Language | Haskell2010 |
Streamly.Internal.Data.MutByteArray
Description
Synopsis
- length :: MutByteArray -> IO Int
- new :: Int -> IO MutByteArray
- empty :: MutByteArray
- type MutableByteArray = MutByteArray
- data MutByteArray = MutByteArray (MutableByteArray# RealWorld)
- data PinnedState
- nil :: MutByteArray
- unsafePutSlice :: MonadIO m => MutByteArray -> Int -> MutByteArray -> Int -> Int -> m ()
- putSliceUnsafe :: MonadIO m => MutByteArray -> Int -> MutByteArray -> Int -> Int -> m ()
- pin :: MutByteArray -> IO MutByteArray
- unpin :: MutByteArray -> IO MutByteArray
- isPinned :: MutByteArray -> Bool
- pinnedNew :: Int -> IO MutByteArray
- unsafeAsPtr :: MonadIO m => MutByteArray -> (Ptr a -> IO b) -> m b
- blockSize :: Int
- asPtrUnsafe :: MonadIO m => MutByteArray -> (Ptr a -> m b) -> m b
- unsafePinnedAsPtr :: MonadIO m => MutByteArray -> (Ptr a -> m b) -> m b
- getMutByteArray# :: MutByteArray -> MutableByteArray# RealWorld
- largeObjectThreshold :: Int
- unsafeByteCmp :: MutByteArray -> Int -> MutByteArray -> Int -> Int -> IO Int
- touch :: MutByteArray -> IO ()
- newAs :: PinnedState -> Int -> IO MutByteArray
- new' :: Int -> IO MutByteArray
- reallocSliceAs :: PinnedState -> Int -> MutByteArray -> Int -> Int -> IO MutByteArray
- unsafePutPtrN :: MonadIO m => Ptr Word8 -> MutByteArray -> Int -> Int -> m ()
- unsafeCloneSliceAs :: MonadIO m => PinnedState -> Int -> Int -> MutByteArray -> m MutByteArray
- unsafeCloneSlice :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray
- unsafePinnedCloneSlice :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray
- getMutableByteArray# :: MutByteArray -> MutableByteArray# RealWorld
- newBytesAs :: PinnedState -> Int -> IO MutByteArray
- sizeOfMutableByteArray :: MutByteArray -> IO Int
- cloneSliceUnsafeAs :: MonadIO m => PinnedState -> Int -> Int -> MutByteArray -> m MutByteArray
- cloneSliceUnsafe :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray
- pinnedCloneSliceUnsafe :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray
- pinnedNewAlignedBytes :: Int -> Int -> IO MutByteArray
- class Unbox a where
- sizeOf :: Proxy a -> Int
- peekAt :: Int -> MutByteArray -> IO a
- peekByteIndex :: Int -> MutByteArray -> IO a
- pokeAt :: Int -> MutByteArray -> a -> IO ()
- pokeByteIndex :: Int -> MutByteArray -> a -> IO ()
- read :: Unbox a => Peeker a
- data BoundedPtr = BoundedPtr MutByteArray Int Int
- newtype Peeker a = Peeker (Builder BoundedPtr IO a)
- readUnsafe :: Unbox a => Peeker a
- skipByte :: Peeker ()
- runPeeker :: Peeker a -> BoundedPtr -> IO a
- pokeBoundedPtrUnsafe :: Unbox a => a -> BoundedPtr -> IO BoundedPtr
- pokeBoundedPtr :: Unbox a => a -> BoundedPtr -> IO BoundedPtr
- class PeekRep (f :: Type -> Type) where
- class PokeRep (f :: Type -> Type) where
- pokeRep :: f a -> BoundedPtr -> IO BoundedPtr
- class SizeOfRep (f :: Type -> Type) where
- genericSizeOf :: SizeOfRep (Rep a) => Proxy a -> Int
- genericPeekByteIndex :: (Generic a, PeekRep (Rep a)) => MutByteArray -> Int -> IO a
- genericPokeByteIndex :: (Generic a, PokeRep (Rep a)) => MutByteArray -> Int -> a -> IO ()
- data DataType = DataType {}
- deriveUnbox :: Q [Dec] -> Q [Dec]
- data DataCon = DataCon {}
- reifyDataType :: Name -> Q DataType
- class Serialize a where
- addSizeTo :: Int -> a -> Int
- deserializeAt :: Int -> MutByteArray -> Int -> IO (Int, a)
- serializeAt :: Int -> MutByteArray -> a -> IO Int
- c2w :: Char -> Word8
- data SerializeConfig = SerializeConfig {}
- inlineAddSizeTo :: Maybe Inline -> SerializeConfig -> SerializeConfig
- inlineSerializeAt :: Maybe Inline -> SerializeConfig -> SerializeConfig
- inlineDeserializeAt :: Maybe Inline -> SerializeConfig -> SerializeConfig
- deriveSerialize :: Q [Dec] -> Q [Dec]
- deriveSerializeWith :: (SerializeConfig -> SerializeConfig) -> Q [Dec] -> Q [Dec]
- mkRecSerializeExpr :: Name -> SimpleDataCon -> Q Exp
- mkRecDeserializeExpr :: Name -> Name -> Name -> SimpleDataCon -> Q Exp
- mkRecSizeOfExpr :: SimpleDataCon -> Q Exp
- conUpdateFuncDec :: Name -> [Field] -> Q [Dec]
- mkDeserializeKeysDec :: Name -> Name -> SimpleDataCon -> Q [Dec]
- mkDeserializeExprOne :: Name -> SimpleDataCon -> Q Exp
- mkSerializeExprFields :: Name -> [Field] -> Q Exp
- serializeConfig :: SerializeConfig
- encodeConstrNames :: Bool -> SerializeConfig -> SerializeConfig
- encodeRecordFields :: Bool -> SerializeConfig -> SerializeConfig
- data TypeOfType
- typeOfType :: Type -> [DataCon] -> TypeOfType
- data SimpleDataCon = SimpleDataCon Name [Field]
- simplifyDataCon :: DataCon -> SimpleDataCon
- type Field = (Maybe Name, Type)
- mkFieldName :: Int -> Name
- isUnitType :: [DataCon] -> Bool
- isRecordSyntax :: SimpleDataCon -> Bool
- wListToString :: [Word8] -> String
- xorCmp :: [Word8] -> Name -> Name -> Q Exp
- serializeW8List :: Name -> Name -> [Word8] -> Q Exp
- litIntegral :: Integral a => a -> Q Exp
- litProxy :: Unbox a => Proxy a -> Q Exp
- matchConstructor :: Name -> Int -> Q Exp -> Q Match
- openConstructor :: Name -> Int -> Q Pat
- makeI :: Int -> Name
- makeN :: Int -> Name
- makeA :: Int -> Name
- int_w8 :: Int -> Word8
- int_w32 :: Int -> Word32
- w32_int :: Word32 -> Int
- w8_int :: Word8 -> Int
- _acc :: Name
- _arr :: Name
- _endOffset :: Name
- _initialOffset :: Name
- _x :: Name
- _tag :: Name
- _val :: Name
- errorUnsupported :: String -> a
- errorUnimplemented :: a
MutByteArray
empty :: MutByteArray Source #
type MutableByteArray = MutByteArray Source #
Deprecated: Please use MutByteArray instead
data MutByteArray Source #
A lifted mutable byte array type wrapping MutableByteArray# RealWorld.
This is a low level array used to back high level unboxed arrays and
serialized data.
Constructors
| MutByteArray (MutableByteArray# RealWorld) |
data PinnedState Source #
Instances
| Show PinnedState Source # | |
Defined in Streamly.Internal.Data.MutByteArray.Type Methods showsPrec :: Int -> PinnedState -> ShowS # show :: PinnedState -> String # showList :: [PinnedState] -> ShowS # | |
| Eq PinnedState Source # | |
Defined in Streamly.Internal.Data.MutByteArray.Type | |
nil :: MutByteArray Source #
Deprecated: Please use empty instead
unsafePutSlice :: MonadIO m => MutByteArray -> Int -> MutByteArray -> Int -> Int -> m () Source #
unsafePutSlice src srcOffset dst dstOffset len copies len bytes from
src at srcOffset to dst at dstOffset.
This is unsafe as it does not check the bounds of src or dst.
putSliceUnsafe :: MonadIO m => MutByteArray -> Int -> MutByteArray -> Int -> Int -> m () Source #
Deprecated: Please use unsafePutSlice instead.
unsafePutSlice src srcOffset dst dstOffset len copies len bytes from
src at srcOffset to dst at dstOffset.
This is unsafe as it does not check the bounds of src or dst.
pin :: MutByteArray -> IO MutByteArray Source #
Return a copy of the array in pinned memory if unpinned, else return the original array.
unpin :: MutByteArray -> IO MutByteArray Source #
Return a copy of the array in unpinned memory if pinned, else return the original array.
unsafeAsPtr :: MonadIO m => MutByteArray -> (Ptr a -> IO b) -> m b Source #
Use a MutByteArray as Ptr a. This is useful when we want to pass
an array as a pointer to some operating system call or to a "safe" FFI call.
Unsafe WARNING:
- Will lead to memory corruption if the array is not pinned. Use only if the array is known to be pinned already or pin it explicitly.
- Ensure that the pointer is accessed within the legal bounds of the array. The size of the MutByteArray must be taken into account.
Pre-release
The page or block size used by the GHC allocator. Allocator allocates at least a block and then allocates smaller allocations from within a block.
asPtrUnsafe :: MonadIO m => MutByteArray -> (Ptr a -> m b) -> m b Source #
Deprecated: Pin the array and then use unsafeAsPtr.
unsafePinnedAsPtr :: MonadIO m => MutByteArray -> (Ptr a -> m b) -> m b Source #
Deprecated: Pin the array and then use unsafeAsPtr.
NOTE: this is deprecated because it can lead to accidental problems if the user tries to use it to mutate the array because it does not return the new array after pinning.
largeObjectThreshold :: Int Source #
Allocations larger than largeObjectThreshold are in multiples of block
size and are always pinned. The space beyond the end of a large object up to
the end of the block is unused.
unsafeByteCmp :: MutByteArray -> Int -> MutByteArray -> Int -> Int -> IO Int Source #
touch :: MutByteArray -> IO () Source #
newAs :: PinnedState -> Int -> IO MutByteArray Source #
reallocSliceAs :: PinnedState -> Int -> MutByteArray -> Int -> Int -> IO MutByteArray Source #
reallocSliceAs pinType newLen array offset len reallocates a slice
from array starting at offset and having length len to a new array of
length newLen copying the old data to the new. Note that if the newLen
is smaller than len it will truncate the old data.
unsafePutPtrN :: MonadIO m => Ptr Word8 -> MutByteArray -> Int -> Int -> m () Source #
unsafePutPtrN srcPtr dst dstOffset len copies len bytes from srcPtr
to dst at dstOffset.
Unsafe:
The caller has to ensure that:
- the MutByteArray
dstis valid up todstOffset + len. - the
srcPtris alive and pinned during the call. - the
srcPtris valid up to lengthlen.
unsafeCloneSliceAs :: MonadIO m => PinnedState -> Int -> Int -> MutByteArray -> m MutByteArray Source #
Unsafe as it does not check whether the start offset and length supplied are valid inside the array.
unsafeCloneSlice :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray Source #
unsafeCloneSlice offset len arr clones a slice of the supplied array
starting at the given offset and equal to the given length.
unsafePinnedCloneSlice :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray Source #
unsafePinnedCloneSlice offset len arr
getMutableByteArray# :: MutByteArray -> MutableByteArray# RealWorld Source #
Deprecated: Please use getMutByteArray# instead.
newBytesAs :: PinnedState -> Int -> IO MutByteArray Source #
Deprecated: Please use newAs instead.
sizeOfMutableByteArray :: MutByteArray -> IO Int Source #
Deprecated: Please use length instead.
Return the size of the array in bytes.
cloneSliceUnsafeAs :: MonadIO m => PinnedState -> Int -> Int -> MutByteArray -> m MutByteArray Source #
Deprecated: Please use unsafeCloneSliceAs instead.
Unsafe as it does not check whether the start offset and length supplied are valid inside the array.
cloneSliceUnsafe :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray Source #
Deprecated: Please use unsafeCloneSlice instead.
unsafeCloneSlice offset len arr clones a slice of the supplied array
starting at the given offset and equal to the given length.
pinnedCloneSliceUnsafe :: MonadIO m => Int -> Int -> MutByteArray -> m MutByteArray Source #
Deprecated: Please use unsafePinnedCloneSlice instead.
unsafePinnedCloneSlice offset len arr
pinnedNewAlignedBytes :: Int -> Int -> IO MutByteArray Source #
Deprecated: Please use pinnedNew instead
Unbox
The Unbox type class provides operations for serialization (unboxing)
and deserialization (boxing) of fixed-length, non-recursive Haskell data
types to and from their byte stream representation.
Unbox uses fixed size encoding, therefore, size is independent of the value,
it must be determined solely by the type. This restriction makes types with
Unbox instances suitable for storing in arrays. Note that sum types may
have multiple constructors of different sizes, the size of a sum type is
computed as the maximum required by any constructor.
The peekAt operation reads as many bytes from the mutable byte
array as the size of the data type and builds a Haskell data type from
these bytes. pokeAt operation converts a Haskell data type to its
binary representation which consists of size bytes and then stores
these bytes into the mutable byte array. These operations do not check the
bounds of the array, the user of the type class is expected to check the
bounds before peeking or poking.
IMPORTANT: The serialized data's byte ordering remains the same as the host machine's byte order. Therefore, it can not be deserialized from host machines with a different byte ordering.
Instances can be derived via Generics, Template Haskell, or written manually. Note that the data type must be non-recursive. WARNING! Generic and Template Haskell deriving, both hang for recursive data types. Deriving via Generics is more convenient but Template Haskell should be preferred over Generics for the following reasons:
- Instances derived via Template Haskell provide better and more reliable performance.
- Generic deriving allows only 256 fields or constructor tags whereas template Haskell has no limit.
Here is an example, for deriving an instance of this type class using generics:
>>>import GHC.Generics (Generic)>>>:{data Object = Object { _int0 :: Int , _int1 :: Int } deriving Generic :}
>>>import Streamly.Data.MutByteArray (Unbox(..))>>>instance Unbox Object
To derive the instance via Template Haskell:
import Streamly.Data.MutByteArray (deriveUnbox) $(deriveUnbox [d|instance Unbox Object|])
See deriveUnbox for more information on deriving
using Template Haskell.
If you want to write the instance manually:
>>>:{instance Unbox Object where sizeOf _ = 16 peekAt i arr = do -- Check the array bounds x0 <- peekAt i arr x1 <- peekAt (i + 8) arr return $ Object x0 x1 pokeAt i arr (Object x0 x1) = do -- Check the array bounds pokeAt i arr x0 pokeAt (i + 8) arr x1 :}
Minimal complete definition
Nothing
Methods
sizeOf :: Proxy a -> Int Source #
Get the size. Size cannot be zero, should be at least 1 byte.
peekAt :: Int -> MutByteArray -> IO a Source #
peekAt byte-offset array reads an element of type a from the
given byte offset in the array.
IMPORTANT: The implementation of this interface may not check the bounds of the array, the caller must not assume that.
peekByteIndex :: Int -> MutByteArray -> IO a Source #
Deprecated: Use peekAt.
pokeAt :: Int -> MutByteArray -> a -> IO () Source #
pokeAt byte-offset array writes an element of type a to the
given byte offset in the array.
IMPORTANT: The implementation of this interface may not check the bounds of the array, the caller must not assume that.
pokeByteIndex :: Int -> MutByteArray -> a -> IO () Source #
Deprecated: Use pokeAt.
Instances
data BoundedPtr Source #
A location inside a mutable byte array with the bound of the array. Is it cheaper to just get the bound using the size of the array whenever needed?
Constructors
| BoundedPtr MutByteArray Int Int |
Chains peek functions that pass the current position to the next function
Constructors
| Peeker (Builder BoundedPtr IO a) |
readUnsafe :: Unbox a => Peeker a Source #
pokeBoundedPtrUnsafe :: Unbox a => a -> BoundedPtr -> IO BoundedPtr Source #
pokeBoundedPtr :: Unbox a => a -> BoundedPtr -> IO BoundedPtr Source #
class PeekRep (f :: Type -> Type) where Source #
Instances
| PeekRep (U1 :: Type -> Type) Source # | |
| PeekRep (V1 :: Type -> Type) Source # | |
| (PeekRep f, PeekRep g) => PeekRep (f :*: g) Source # | |
| (MaxArity256 (SumArity (f :+: g)), KnownNat (SumArity (f :+: g)), PeekRepSum 0 (f :+: g)) => PeekRep (f :+: g) Source # | |
| Unbox a => PeekRep (K1 i a :: Type -> Type) Source # | |
| PeekRep f => PeekRep (M1 i c f) Source # | |
class PokeRep (f :: Type -> Type) where Source #
Methods
pokeRep :: f a -> BoundedPtr -> IO BoundedPtr Source #
Instances
| PokeRep (U1 :: Type -> Type) Source # | |
Defined in Streamly.Internal.Data.Unbox Methods pokeRep :: U1 a -> BoundedPtr -> IO BoundedPtr Source # | |
| PokeRep (V1 :: Type -> Type) Source # | |
Defined in Streamly.Internal.Data.Unbox Methods pokeRep :: V1 a -> BoundedPtr -> IO BoundedPtr Source # | |
| (PokeRep f, PokeRep g) => PokeRep (f :*: g) Source # | |
Defined in Streamly.Internal.Data.Unbox Methods pokeRep :: (f :*: g) a -> BoundedPtr -> IO BoundedPtr Source # | |
| (MaxArity256 (SumArity (f :+: g)), PokeRepSum 0 (f :+: g)) => PokeRep (f :+: g) Source # | |
Defined in Streamly.Internal.Data.Unbox Methods pokeRep :: (f :+: g) a -> BoundedPtr -> IO BoundedPtr Source # | |
| Unbox a => PokeRep (K1 i a :: Type -> Type) Source # | |
Defined in Streamly.Internal.Data.Unbox Methods pokeRep :: K1 i a a0 -> BoundedPtr -> IO BoundedPtr Source # | |
| PokeRep f => PokeRep (M1 i c f) Source # | |
Defined in Streamly.Internal.Data.Unbox Methods pokeRep :: M1 i c f a -> BoundedPtr -> IO BoundedPtr Source # | |
class SizeOfRep (f :: Type -> Type) where Source #
Implementation of sizeOf that works on the generic representation of an ADT.
Instances
| SizeOfRep (U1 :: Type -> Type) Source # | |
| SizeOfRep (V1 :: Type -> Type) Source # | |
| (SizeOfRep f, SizeOfRep g) => SizeOfRep (f :*: g) Source # | |
| (MaxArity256 (SumArity (f :+: g)), SizeOfRepSum f, SizeOfRepSum g) => SizeOfRep (f :+: g) Source # | |
| Unbox a => SizeOfRep (K1 i a :: Type -> Type) Source # | |
| SizeOfRep f => SizeOfRep (M1 i c f) Source # | |
genericPeekByteIndex :: (Generic a, PeekRep (Rep a)) => MutByteArray -> Int -> IO a Source #
genericPokeByteIndex :: (Generic a, PokeRep (Rep a)) => MutByteArray -> Int -> a -> IO () Source #
Simplified info about a DataD. Omits deriving, strictness,
kind info, and whether it's data or newtype.
deriveUnbox :: Q [Dec] -> Q [Dec] Source #
Given an Unbox instance declaration splice without the methods (e.g.
[d|instance Unbox a => Unbox (Maybe a)|]), generate an instance
declaration including all the type class method implementations.
Usage:
$(deriveUnbox [d|instance Unbox a => Unbox (Maybe a)|])
Simplified info about a Con. Omits deriving, strictness, and kind
info. This is much nicer than consuming Con directly, because it
unifies all the constructors into one.
Constructors
| DataCon | |
reifyDataType :: Name -> Q DataType Source #
Reify the given data or newtype declaration, and yields its
DataType representation.
Serialize
class Serialize a where Source #
The Serialize type class provides operations for serialization and
deserialization of general Haskell data types to and from their byte stream
representation.
Unlike Unbox, Serialize uses variable length encoding, therefore, it can
serialize recursive and variable length data types like lists, or variable
length sum types where the length of the value may vary depending on a
particular constructor. For variable length data types the length is encoded
along with the data.
The deserializeAt operation reads bytes from the mutable byte array and
builds a Haskell data type from these bytes, the number of bytes it reads
depends on the type and the encoded value it is reading. serializeAt
operation converts a Haskell data type to its binary representation which
must consist of as many bytes as added by the addSizeTo operation for that
value and then stores these bytes into the mutable byte array. The
programmer is expected to use the addSizeTo operation and allocate an
array of sufficient length before calling serializeAt.
IMPORTANT: The serialized data's byte ordering remains the same as the host machine's byte order. Therefore, it can not be deserialized from host machines with a different byte ordering.
Instances can be derived via Template Haskell, or written manually.
Here is an example, for deriving an instance of this type class using template Haskell:
>>>:{data Object = Object { _obj1 :: [Int] , _obj2 :: Int } :}
import Streamly.Data.MutByteArray (deriveSerialize) $(deriveSerialize [d|instance Serialize Object|])
See deriveSerialize and
deriveSerializeWith for more information on
deriving using Template Haskell.
Here is an example of a manual instance.
>>>import Streamly.Data.MutByteArray (Serialize(..))
>>>:{instance Serialize Object where addSizeTo acc obj = addSizeTo (addSizeTo acc (_obj1 obj)) (_obj2 obj) deserializeAt i arr len = do -- Check the array bounds before reading (i1, x0) <- deserializeAt i arr len (i2, x1) <- deserializeAt i1 arr len pure (i2, Object x0 x1) serializeAt i arr (Object x0 x1) = do i1 <- serializeAt i arr x0 i2 <- serializeAt i1 arr x1 pure i2 :}
Methods
addSizeTo :: Int -> a -> Int Source #
addSizeTo accum value returns accum incremented by the size of the
serialized representation of value in bytes. Size cannot be zero. It
should be at least 1 byte.
deserializeAt :: Int -> MutByteArray -> Int -> IO (Int, a) Source #
deserializeAt byte-offset array arrayLen deserializes a value from
the given byte-offset in the array. Returns a tuple consisting of the
next byte-offset and the deserialized value.
The arrayLen passed is the entire length of the input buffer. It is to be used to check if we would overflow the input buffer when deserializing.
Throws an exception if the operation would exceed the supplied arrayLen.
serializeAt :: Int -> MutByteArray -> a -> IO Int Source #
serializeAt byte-offset array value writes the serialized
representation of the value in the array at the given byte-offset.
Returns the next byte-offset.
This is an unsafe operation, the programmer must ensure that the array
has enough space available to serialize the value as determined by the
addSizeTo operation.
Instances
Serialize TH
data SerializeConfig Source #
Configuration to control how the Serialize instance is generated. The
configuration is opaque and is modified by composing config modifier
functions, for example:
>>>import Language.Haskell.TH (Inline(..))>>>configOpts = (inlineDeserializeAt (Just NoInline)) . (inlineSerializeAt (Just Inlinable))
The default configuration settings are:
inlineAddSizeToNothinginlineSerializeAt(Just Inline)inlineDeserializeAt(Just Inline)
The following experimental options are also available:
encodeConstrNamesFalseencodeRecordFieldsFalse
Constructors
| SerializeConfig | |
inlineSerializeAt :: Maybe Inline -> SerializeConfig -> SerializeConfig Source #
How should we inline the serialize function? The default 'Just Inline'.
However, aggressive inlining can bloat the code and increase in compilation
times when there are big functions and too many nesting levels so you can
change it accordingly. A Nothing value leaves the decision to the
compiler.
inlineDeserializeAt :: Maybe Inline -> SerializeConfig -> SerializeConfig Source #
How should we inline the deserialize function? See guidelines in
inlineSerializeAt.
deriveSerialize :: Q [Dec] -> Q [Dec] Source #
Given an Serialize instance declaration splice without the methods (e.g.
[d|instance Serialize a => Serialize (Maybe a)|]), generate an instance
declaration including all the type class method implementations.
>>>deriveSerialize = deriveSerializeWith id
Usage:
$(deriveSerialize
[d|instance Serialize a => Serialize (Maybe a)|])
deriveSerializeWith :: (SerializeConfig -> SerializeConfig) -> Q [Dec] -> Q [Dec] Source #
deriveSerializeWith config-modifier instance-dec generates a template
Haskell splice consisting of a declaration of a Serialize instance.
instance-dec is a template Haskell declaration splice consisting of a
standard Haskell instance declaration without the type class methods (e.g.
[d|instance Serialize a => Serialize (Maybe a)|]).
The type class methods for the given instance are generated according to the
supplied config-modifier parameter. See SerializeConfig for default
configuration settings.
Usage:
$(deriveSerializeWith
( inlineSerializeAt (Just NoInline)
. inlineDeserializeAt (Just NoInline)
)
[d|instance Serialize a => Serialize (Maybe a)|])
mkRecSerializeExpr :: Name -> SimpleDataCon -> Q Exp Source #
mkRecDeserializeExpr :: Name -> Name -> Name -> SimpleDataCon -> Q Exp Source #
mkRecSizeOfExpr :: SimpleDataCon -> Q Exp Source #
mkDeserializeKeysDec :: Name -> Name -> SimpleDataCon -> Q [Dec] Source #
mkDeserializeExprOne :: Name -> SimpleDataCon -> Q Exp Source #
encodeConstrNames :: Bool -> SerializeConfig -> SerializeConfig Source #
Experimental
In sum types, use Latin-1 encoded original constructor names rather than binary values to identify constructors. This option is not applicable to product types.
This option enables the following behavior:
- Reordering: Order of the fields can be changed without affecting serialization.
- Addition: If a field is added in the new version, the old version of the data type can still be deserialized by the new version. The new value would never occur in the old one.
- Deletion: If a field is deleted in the new version, deserialization of the old version will result in an error. TBD: We can possibly designate a catch-all case to handle this scenario.
Note that if you change a type, change the semantics of a type, or delete a field and add a new field with the same name, deserialization of old data may result in silent unexpected behavior.
This option has to be the same on both encoding and decoding side.
The default is False.
encodeRecordFields :: Bool -> SerializeConfig -> SerializeConfig Source #
Experimental
In explicit record types, use Latin-1 encoded record field names rather than binary values to identify the record fields. Note that this option is not applicable to sum types. Also, it does not work on a product type which is not a record, because there are no field names to begin with.
This option enables the following behavior:
- Reordering: Order of the fields can be changed without affecting serialization.
- Addition: If a
Maybetype field is added in the new version, the old version of the data type can still be deserialized by the new version, the field value in the older version is assumed to beNothing. If any other type of field is added, deserialization of the older version results in an error but only when that field is actually accessed in the deserialized record. - Deletion: If a field is deleted in the new version and it is encountered in a previously serialized version then the field is discarded.
This option has to be the same on both encoding and decoding side.
There is a constant performance overhead proportional to the total length of the record field names and the number of record fields.
The default is False.
data TypeOfType Source #
Constructors
| UnitType Name | |
| TheType SimpleDataCon | |
| MultiType [SimpleDataCon] |
Instances
typeOfType :: Type -> [DataCon] -> TypeOfType Source #
data SimpleDataCon Source #
Constructors
| SimpleDataCon Name [Field] |
Instances
| Eq SimpleDataCon Source # | |
Defined in Streamly.Internal.Data.Serialize.TH.Bottom Methods (==) :: SimpleDataCon -> SimpleDataCon -> Bool # (/=) :: SimpleDataCon -> SimpleDataCon -> Bool # | |
mkFieldName :: Int -> Name Source #
isUnitType :: [DataCon] -> Bool Source #
isRecordSyntax :: SimpleDataCon -> Bool Source #
wListToString :: [Word8] -> String Source #
_endOffset :: Name Source #
errorUnsupported :: String -> a Source #
errorUnimplemented :: a Source #