{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE CPP #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE MultiWayIf #-}
{-# LANGUAGE ScopedTypeVariables #-}
module Flat.Encoder.Prim
(
eBits16F
, eBitsF
, eFloatF
, eDoubleF
#if ! defined (ETA_VERSION)
, eUTF16F
#endif
, eUTF8F
, eCharF
, eNaturalF
, eIntegerF
, eInt64F
, eInt32F
, eIntF
, eInt16F
, eInt8F
, eWordF
, eWord64F
, eWord32F
, eWord16F
, eBytesF
, eLazyBytesF
, eShortBytesF
, eWord8F
, eFillerF
, eBoolF
, eTrueF
, eFalseF
, varWordF
, updateWord8
, w7l
, eWord32BEF
, eWord64BEF
, eWord32E
, eWord64E
) where
import Control.Monad
import qualified Data.ByteString as B
import qualified Data.ByteString.Lazy as L
import qualified Data.ByteString.Lazy.Internal as L
import qualified Data.ByteString.Short.Internal as SBS
import Data.Char
import Data.FloatCast
import Data.Primitive.ByteArray
import qualified Data.Text as T
import Flat.Encoder.Types
import Flat.Endian
import Flat.Memory
import Flat.Types
#if ! defined (ETA_VERSION) && ! MIN_VERSION_text(2,0,0)
import qualified Data.Text.Array as TA
import qualified Data.Text.Internal as TI
#endif
import qualified Data.Text.Encoding as TE
import Data.ZigZag
import Foreign
#include "MachDeps.h"
{-# INLINE eFloatF #-}
eFloatF :: Float -> Prim
eFloatF :: Float -> Prim
eFloatF = Word32 -> Prim
eWord32BEF forall b c a. (b -> c) -> (a -> b) -> a -> c
. Float -> Word32
floatToWord
{-# INLINE eDoubleF #-}
eDoubleF :: Double -> Prim
eDoubleF :: Double -> Prim
eDoubleF = Word64 -> Prim
eWord64BEF forall b c a. (b -> c) -> (a -> b) -> a -> c
. Double -> Word64
doubleToWord
{-# INLINE eWord64BEF #-}
eWord64BEF :: Word64 -> Prim
eWord64BEF :: Word64 -> Prim
eWord64BEF = (Word64 -> Word64) -> Word64 -> Prim
eWord64E Word64 -> Word64
toBE64
{-# INLINE eWord32BEF #-}
eWord32BEF :: Word32 -> Prim
eWord32BEF :: Word32 -> Prim
eWord32BEF = (Word32 -> Word32) -> Word32 -> Prim
eWord32E Word32 -> Word32
toBE32
{-# INLINE eCharF #-}
eCharF :: Char -> Prim
eCharF :: Char -> Prim
eCharF = Word32 -> Prim
eWord32F forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a b. (Integral a, Num b) => a -> b
fromIntegral forall b c a. (b -> c) -> (a -> b) -> a -> c
. Char -> Int
ord
{-# INLINE eWordF #-}
eWordF :: Word -> Prim
{-# INLINE eIntF #-}
eIntF :: Int -> Prim
#if WORD_SIZE_IN_BITS == 64
eWordF :: Word -> Prim
eWordF = Word64 -> Prim
eWord64F forall b c a. (b -> c) -> (a -> b) -> a -> c
. (forall a b. (Integral a, Num b) => a -> b
fromIntegral :: Word -> Word64)
eIntF :: Int -> Prim
eIntF = Int64 -> Prim
eInt64F forall b c a. (b -> c) -> (a -> b) -> a -> c
. (forall a b. (Integral a, Num b) => a -> b
fromIntegral :: Int -> Int64)
#elif WORD_SIZE_IN_BITS == 32
eWordF = eWord32F . (fromIntegral :: Word -> Word32)
eIntF = eInt32F . (fromIntegral :: Int -> Int32)
#else
#error expected WORD_SIZE_IN_BITS to be 32 or 64
#endif
{-# INLINE eInt8F #-}
eInt8F :: Int8 -> Prim
eInt8F :: Int8 -> Prim
eInt8F = Word8 -> Prim
eWord8F forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall signed unsigned.
ZigZag signed unsigned =>
signed -> unsigned
zigZag
{-# INLINE eInt16F #-}
eInt16F :: Int16 -> Prim
eInt16F :: Int16 -> Prim
eInt16F = Word16 -> Prim
eWord16F forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall signed unsigned.
ZigZag signed unsigned =>
signed -> unsigned
zigZag
{-# INLINE eInt32F #-}
eInt32F :: Int32 -> Prim
eInt32F :: Int32 -> Prim
eInt32F = Word32 -> Prim
eWord32F forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall signed unsigned.
ZigZag signed unsigned =>
signed -> unsigned
zigZag
{-# INLINE eInt64F #-}
eInt64F :: Int64 -> Prim
eInt64F :: Int64 -> Prim
eInt64F = Word64 -> Prim
eWord64F forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall signed unsigned.
ZigZag signed unsigned =>
signed -> unsigned
zigZag
{-# INLINE eIntegerF #-}
eIntegerF :: Integer -> Prim
eIntegerF :: Integer -> Prim
eIntegerF = forall t. (Bits t, Integral t) => t -> Prim
eIntegralF forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall signed unsigned.
ZigZag signed unsigned =>
signed -> unsigned
zigZag
{-# INLINE eNaturalF #-}
eNaturalF :: Natural -> Prim
eNaturalF :: Natural -> Prim
eNaturalF = forall t. (Bits t, Integral t) => t -> Prim
eIntegralF forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. Integral a => a -> Integer
toInteger
{-# INLINE eIntegralF #-}
eIntegralF :: (Bits t, Integral t) => t -> Prim
eIntegralF :: forall t. (Bits t, Integral t) => t -> Prim
eIntegralF t
t =
let vs :: [Word8]
vs = forall t. (Bits t, Integral t) => t -> [Word8]
w7l t
t
in [Word8] -> Prim
eIntegralW [Word8]
vs
w7l :: (Bits t, Integral t) => t -> [Word8]
w7l :: forall t. (Bits t, Integral t) => t -> [Word8]
w7l t
t =
let l :: Word8
l = forall a. Integral a => a -> Word8
low7 t
t
t' :: t
t' = t
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
7
in if t
t' forall a. Eq a => a -> a -> Bool
== t
0
then [Word8
l]
else Word8 -> Word8
w7 Word8
l forall a. a -> [a] -> [a]
: forall t. (Bits t, Integral t) => t -> [Word8]
w7l t
t'
where
{-# INLINE w7 #-}
w7 :: Word8 -> Word8
w7 :: Word8 -> Word8
w7 Word8
l = Word8
l forall a. Bits a => a -> a -> a
.|. Word8
0x80
{-# INLINE eIntegralW #-}
eIntegralW :: [Word8] -> Prim
eIntegralW :: [Word8] -> Prim
eIntegralW [Word8]
vs s :: S
s@(S Ptr Word8
op Word8
_ Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
0 = forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m b
foldM forall a b. Storable a => Ptr a -> a -> IO (Ptr b)
pokeWord' Ptr Word8
op [Word8]
vs forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \Ptr Word8
op' -> forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S Ptr Word8
op' Word8
0 Int
0)
| Bool
otherwise = forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m b
foldM (forall a b c. (a -> b -> c) -> b -> a -> c
flip Word8 -> Prim
eWord8F) S
s [Word8]
vs
{-# INLINE eWord8F #-}
eWord8F :: Word8 -> Prim
eWord8F :: Word8 -> Prim
eWord8F Word8
t s :: S
s@(S Ptr Word8
op Word8
_ Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
0 = forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op Word8
t
| Bool
otherwise = Word8 -> Prim
eByteUnaligned Word8
t S
s
{-# INLINE eWord32E #-}
eWord32E :: (Word32 -> Word32) -> Word32 -> Prim
eWord32E :: (Word32 -> Word32) -> Word32 -> Prim
eWord32E Word32 -> Word32
conv Word32
t (S Ptr Word8
op Word8
w Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
0 = forall a t a1. Storable a => (t -> a) -> Ptr a1 -> t -> IO ()
pokeW Word32 -> Word32
conv Ptr Word8
op Word32
t forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> forall (m :: * -> *) a. Monad m => Ptr a -> Int -> m S
skipBytes Ptr Word8
op Int
4
| Bool
otherwise =
forall a t a1. Storable a => (t -> a) -> Ptr a1 -> t -> IO ()
pokeW Word32 -> Word32
conv Ptr Word8
op (forall a. Integral a => a -> Word32
asWord32 Word8
w forall a. Bits a => a -> Int -> a
`unsafeShiftL` Int
24 forall a. Bits a => a -> a -> a
.|. Word32
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o) forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>>
forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr Word8
op Int
4) (forall a. Integral a => a -> Word8
asWord8 Word32
t forall a. Bits a => a -> Int -> a
`unsafeShiftL` (Int
8 forall a. Num a => a -> a -> a
- Int
o)) Int
o)
{-# INLINE eWord64E #-}
eWord64E :: (Word64 -> Word64) -> Word64 -> Prim
eWord64E :: (Word64 -> Word64) -> Word64 -> Prim
eWord64E Word64 -> Word64
conv Word64
t (S Ptr Word8
op Word8
w Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
0 = forall t a. (t -> Word64) -> Ptr a -> t -> IO ()
poke64 Word64 -> Word64
conv Ptr Word8
op Word64
t forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> forall (m :: * -> *) a. Monad m => Ptr a -> Int -> m S
skipBytes Ptr Word8
op Int
8
| Bool
otherwise =
forall t a. (t -> Word64) -> Ptr a -> t -> IO ()
poke64 Word64 -> Word64
conv Ptr Word8
op (forall a. Integral a => a -> Word64
asWord64 Word8
w forall a. Bits a => a -> Int -> a
`unsafeShiftL` Int
56 forall a. Bits a => a -> a -> a
.|. Word64
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o) forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>>
forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr Word8
op Int
8) (forall a. Integral a => a -> Word8
asWord8 Word64
t forall a. Bits a => a -> Int -> a
`unsafeShiftL` (Int
8 forall a. Num a => a -> a -> a
- Int
o)) Int
o)
{-# INLINE eWord16F #-}
eWord16F :: Word16 -> Prim
eWord16F :: Word16 -> Prim
eWord16F = forall t. (Bits t, Integral t) => t -> Prim
varWordF
{-# INLINE eWord32F #-}
eWord32F :: Word32 -> Prim
eWord32F :: Word32 -> Prim
eWord32F = forall t. (Bits t, Integral t) => t -> Prim
varWordF
{-# INLINE eWord64F #-}
eWord64F :: Word64 -> Prim
eWord64F :: Word64 -> Prim
eWord64F = forall t. (Bits t, Integral t) => t -> Prim
varWordF
{-# INLINE varWordF #-}
varWordF :: (Bits t, Integral t) => t -> Prim
varWordF :: forall t. (Bits t, Integral t) => t -> Prim
varWordF t
t s :: S
s@(S Ptr Word8
_ Word8
_ Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
0 = forall t. (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWord Word8 -> Prim
eByteAligned t
t S
s
| Bool
otherwise = forall t. (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWord Word8 -> Prim
eByteUnaligned t
t S
s
{-# INLINE varWord #-}
varWord :: (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWord :: forall t. (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWord Word8 -> Prim
writeByte t
t S
s
| t
t forall a. Ord a => a -> a -> Bool
< t
128 = Word8 -> Prim
writeByte (forall a b. (Integral a, Num b) => a -> b
fromIntegral t
t) S
s
| t
t forall a. Ord a => a -> a -> Bool
< t
16384 = forall {m :: * -> *} {a} {t} {b}.
(Monad m, Integral a, Bits t, Bits a, Num t) =>
(t -> b -> m b) -> a -> b -> m b
varWord2_ Word8 -> Prim
writeByte t
t S
s
| t
t forall a. Ord a => a -> a -> Bool
< t
2097152 = forall {m :: * -> *} {a} {t} {b}.
(Monad m, Integral a, Bits t, Bits a, Num t) =>
(t -> b -> m b) -> a -> b -> m b
varWord3_ Word8 -> Prim
writeByte t
t S
s
| Bool
otherwise = forall t. (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWordN_ Word8 -> Prim
writeByte t
t S
s
where
{-# INLINE varWord2_ #-}
varWord2_ :: (t -> b -> m b) -> a -> b -> m b
varWord2_ t -> b -> m b
writeByte a
t b
s =
t -> b -> m b
writeByte (forall a b. (Integral a, Num b) => a -> b
fromIntegral a
t forall a. Bits a => a -> a -> a
.|. t
0x80) b
s forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>=
t -> b -> m b
writeByte (forall a b. (Integral a, Num b) => a -> b
fromIntegral (a
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
7) forall a. Bits a => a -> a -> a
.&. t
0x7F)
{-# INLINE varWord3_ #-}
varWord3_ :: (t -> b -> m b) -> a -> b -> m b
varWord3_ t -> b -> m b
writeByte a
t b
s =
t -> b -> m b
writeByte (forall a b. (Integral a, Num b) => a -> b
fromIntegral a
t forall a. Bits a => a -> a -> a
.|. t
0x80) b
s forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>=
t -> b -> m b
writeByte (forall a b. (Integral a, Num b) => a -> b
fromIntegral (a
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
7) forall a. Bits a => a -> a -> a
.|. t
0x80) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>=
t -> b -> m b
writeByte (forall a b. (Integral a, Num b) => a -> b
fromIntegral (a
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
14) forall a. Bits a => a -> a -> a
.&. t
0x7F)
varWordN_ :: (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWordN_ :: forall t. (Bits t, Integral t) => (Word8 -> Prim) -> t -> Prim
varWordN_ Word8 -> Prim
writeByte = forall t. (Bits t, Integral t) => t -> Prim
go
where
go :: t -> Prim
go !t
v !S
st =
let !l :: Word8
l = forall a. Integral a => a -> Word8
low7 t
v
!v' :: t
v' = t
v forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
7
in if t
v' forall a. Eq a => a -> a -> Bool
== t
0
then Word8 -> Prim
writeByte Word8
l S
st
else Word8 -> Prim
writeByte (Word8
l forall a. Bits a => a -> a -> a
.|. Word8
0x80) S
st forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= t -> Prim
go t
v'
{-# INLINE low7 #-}
low7 :: (Integral a) => a -> Word8
low7 :: forall a. Integral a => a -> Word8
low7 a
t = forall a b. (Integral a, Num b) => a -> b
fromIntegral a
t forall a. Bits a => a -> a -> a
.&. Word8
0x7F
eUTF8F :: T.Text -> Prim
eUTF8F :: Text -> Prim
eUTF8F = ByteString -> Prim
eBytesF forall b c a. (b -> c) -> (a -> b) -> a -> c
. Text -> ByteString
TE.encodeUtf8
#if ! defined (ETA_VERSION)
eUTF16F :: T.Text -> Prim
#if MIN_VERSION_text(2,0,0)
eUTF16F = eBytesF . TE.encodeUtf16LE
#else
eUTF16F :: Text -> Prim
eUTF16F Text
t = Prim
eFillerF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> Text -> Prim
eUTF16F_ Text
t
where
eUTF16F_ :: Text -> Prim
eUTF16F_ (TI.Text (TA.Array ByteArray#
array) Int
w16Off Int
w16Len) S
s =
ByteArray# -> Int -> Int -> Ptr Word8 -> IO S
writeArray ByteArray#
array (Int
2 forall a. Num a => a -> a -> a
* Int
w16Off) (Int
2 forall a. Num a => a -> a -> a
* Int
w16Len) (S -> Ptr Word8
nextPtr S
s)
#endif
#endif
eLazyBytesF :: L.ByteString -> Prim
eLazyBytesF :: ByteString -> Prim
eLazyBytesF ByteString
bs = Prim
eFillerF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> \S
s -> ByteString -> Ptr Word8 -> IO S
write ByteString
bs (S -> Ptr Word8
nextPtr S
s)
where
write :: ByteString -> Ptr Word8 -> IO S
write ByteString
lbs Ptr Word8
op = do
case ByteString
lbs of
L.Chunk ByteString
h ByteString
t -> ByteString -> Ptr Word8 -> IO (Ptr Word8)
writeBS ByteString
h Ptr Word8
op forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ByteString -> Ptr Word8 -> IO S
write ByteString
t
ByteString
L.Empty -> forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op Word8
0
{-# INLINE eShortBytesF #-}
eShortBytesF :: SBS.ShortByteString -> Prim
eShortBytesF :: ShortByteString -> Prim
eShortBytesF ShortByteString
bs = Prim
eFillerF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> ShortByteString -> Prim
eShortBytesF_ ShortByteString
bs
where
eShortBytesF_ :: SBS.ShortByteString -> Prim
eShortBytesF_ :: ShortByteString -> Prim
eShortBytesF_ bs :: ShortByteString
bs@(SBS.SBS ByteArray#
arr) (S Ptr Word8
op Word8
_ Int
0) = ByteArray# -> Int -> Int -> Ptr Word8 -> IO S
writeArray ByteArray#
arr Int
0 (ShortByteString -> Int
SBS.length ShortByteString
bs) Ptr Word8
op
eShortBytesF_ ShortByteString
_ S
_ = forall a. HasCallStack => [Char] -> a
error [Char]
"impossible"
writeArray :: ByteArray# -> Int -> Int -> Ptr Word8 -> IO S
writeArray :: ByteArray# -> Int -> Int -> Ptr Word8 -> IO S
writeArray ByteArray#
arr Int
soff Int
slen Ptr Word8
sop = do
Ptr Word8
op' <- Int -> Int -> Ptr Word8 -> IO (Ptr Word8)
go Int
soff Int
slen Ptr Word8
sop
forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op' Word8
0
where
go :: Int -> Int -> Ptr Word8 -> IO (Ptr Word8)
go !Int
off !Int
len !Ptr Word8
op
| Int
len forall a. Eq a => a -> a -> Bool
== Int
0 = forall (m :: * -> *) a. Monad m => a -> m a
return Ptr Word8
op
| Bool
otherwise =
let l :: Int
l = forall a. Ord a => a -> a -> a
min Int
255 Int
len
in forall a b. Storable a => Ptr a -> a -> IO (Ptr b)
pokeWord' Ptr Word8
op (forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
l) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ByteArray# -> Int -> Int -> Ptr Word8 -> IO (Ptr Word8)
pokeByteArray ByteArray#
arr Int
off Int
l forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>=
Int -> Int -> Ptr Word8 -> IO (Ptr Word8)
go (Int
off forall a. Num a => a -> a -> a
+ Int
l) (Int
len forall a. Num a => a -> a -> a
- Int
l)
eBytesF :: B.ByteString -> Prim
eBytesF :: ByteString -> Prim
eBytesF ByteString
bs = Prim
eFillerF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> Prim
eBytesF_
where
eBytesF_ :: Prim
eBytesF_ S
s = do
Ptr Word8
op' <- ByteString -> Ptr Word8 -> IO (Ptr Word8)
writeBS ByteString
bs (S -> Ptr Word8
nextPtr S
s)
forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op' Word8
0
{-# INLINE eBits16F #-}
eBits16F :: NumBits -> Word16 -> Prim
eBits16F :: Int -> Word16 -> Prim
eBits16F Int
9 Word16
code =
Int -> Word8 -> Prim
eBitsF Int
1 (forall a b. (Integral a, Num b) => a -> b
fromIntegral forall a b. (a -> b) -> a -> b
$ Word16
code forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
8) forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=>
Int -> Word8 -> Prim
eBitsF_ Int
8 (forall a b. (Integral a, Num b) => a -> b
fromIntegral Word16
code)
eBits16F Int
numBits Word16
code = Int -> Word8 -> Prim
eBitsF Int
numBits (forall a b. (Integral a, Num b) => a -> b
fromIntegral Word16
code)
{-# INLINE eBitsF #-}
eBitsF :: NumBits -> Word8 -> Prim
eBitsF :: Int -> Word8 -> Prim
eBitsF Int
1 Word8
0 = Prim
eFalseF
eBitsF Int
1 Word8
1 = Prim
eTrueF
eBitsF Int
2 Word8
0 = Prim
eFalseF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> Prim
eFalseF
eBitsF Int
2 Word8
1 = Prim
eFalseF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> Prim
eTrueF
eBitsF Int
2 Word8
2 = Prim
eTrueF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> Prim
eFalseF
eBitsF Int
2 Word8
3 = Prim
eTrueF forall (m :: * -> *) a b c.
Monad m =>
(a -> m b) -> (b -> m c) -> a -> m c
>=> Prim
eTrueF
eBitsF Int
n Word8
t = Int -> Word8 -> Prim
eBitsF_ Int
n Word8
t
eBitsF_ :: NumBits -> Word8 -> Prim
eBitsF_ :: Int -> Word8 -> Prim
eBitsF_ Int
n Word8
t (S Ptr Word8
op Word8
w Int
o) =
let o' :: Int
o' = Int
o forall a. Num a => a -> a -> a
+ Int
n
f :: Int
f = Int
8 forall a. Num a => a -> a -> a
- Int
o'
in if | Int
f forall a. Ord a => a -> a -> Bool
> Int
0 -> forall (m :: * -> *) a. Monad m => a -> m a
return forall a b. (a -> b) -> a -> b
$ Ptr Word8 -> Word8 -> Int -> S
S Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftL` Int
f)) Int
o'
| Int
f forall a. Eq a => a -> a -> Bool
== Int
0 -> forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. Word8
t)
| Bool
otherwise ->
let o'' :: Int
o'' = -Int
f
in forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o'')) forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>>
forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr Word8
op Int
1) (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftL` (Int
8 forall a. Num a => a -> a -> a
- Int
o'')) Int
o'')
{-# INLINE eBoolF #-}
eBoolF :: Bool -> Prim
eBoolF :: Bool -> Prim
eBoolF Bool
False = Prim
eFalseF
eBoolF Bool
True = Prim
eTrueF
{-# INLINE eTrueF #-}
eTrueF :: Prim
eTrueF :: Prim
eTrueF (S Ptr Word8
op Word8
w Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
7 = forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. Word8
1)
| Bool
otherwise = forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. Word8
128 forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o) (Int
o forall a. Num a => a -> a -> a
+ Int
1))
{-# INLINE eFalseF #-}
eFalseF :: Prim
eFalseF :: Prim
eFalseF (S Ptr Word8
op Word8
w Int
o)
| Int
o forall a. Eq a => a -> a -> Bool
== Int
7 = forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op Word8
w
| Bool
otherwise = forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S Ptr Word8
op Word8
w (Int
o forall a. Num a => a -> a -> a
+ Int
1))
{-# INLINE eFillerF #-}
eFillerF :: Prim
eFillerF :: Prim
eFillerF (S Ptr Word8
op Word8
w Int
_) = forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. Word8
1)
{-# INLINE eByteUnaligned #-}
eByteUnaligned :: Word8 -> Prim
eByteUnaligned :: Word8 -> Prim
eByteUnaligned Word8
t (S Ptr Word8
op Word8
w Int
o) =
forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o)) forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>>
forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr Word8
op Int
1) (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftL` (Int
8 forall a. Num a => a -> a -> a
- Int
o)) Int
o)
{-# INLINE eByteAligned #-}
eByteAligned :: Word8 -> Prim
eByteAligned :: Word8 -> Prim
eByteAligned Word8
t (S Ptr Word8
op Word8
_ Int
_) = forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr Word8
op Word8
t
updateWord8 :: Word8 -> S -> Prim
updateWord8 :: Word8 -> S -> Prim
updateWord8 Word8
t S
mem S
s = do
S -> IO ()
uncache S
s
Word8 -> S -> IO ()
pokeWord8 Word8
t S
mem
Prim
cache S
s
uncache :: S -> IO ()
uncache :: S -> IO ()
uncache S
s = forall a. Storable a => Ptr a -> a -> IO ()
poke (S -> Ptr Word8
nextPtr S
s) (S -> Word8
currByte S
s)
cache :: Prim
cache :: Prim
cache S
s = do
Word8
w <- (S -> Word8
mask S
s forall a. Bits a => a -> a -> a
.&.) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> forall a. Storable a => Ptr a -> IO a
peek (S -> Ptr Word8
nextPtr S
s)
forall (m :: * -> *) a. Monad m => a -> m a
return forall a b. (a -> b) -> a -> b
$ S
s {currByte :: Word8
currByte = Word8
w}
mask :: S -> Word8
mask :: S -> Word8
mask S
s = Word8
255 forall a. Bits a => a -> Int -> a
`unsafeShiftL` (Int
8 forall a. Num a => a -> a -> a
- S -> Int
usedBits S
s)
{-# INLINE pokeWord8 #-}
pokeWord8 :: Word8 -> S -> IO ()
pokeWord8 :: Word8 -> S -> IO ()
pokeWord8 Word8
t (S Ptr Word8
op Word8
_ Int
0) = forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr Word8
op Word8
t
pokeWord8 Word8
t (S Ptr Word8
op Word8
w Int
o) = do
forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr Word8
op (Word8
w forall a. Bits a => a -> a -> a
.|. (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o))
let Ptr Word8
op' :: Ptr Word8 = forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr Word8
op Int
1
Word8
v :: Word8 <- forall a. Storable a => Ptr a -> IO a
peek Ptr Word8
op'
forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr Word8
op' (Word8
t forall a. Bits a => a -> Int -> a
`unsafeShiftL` (Int
8 forall a. Num a => a -> a -> a
- Int
o) forall a. Bits a => a -> a -> a
.|. ((Word8
v forall a. Bits a => a -> Int -> a
`unsafeShiftL` Int
o) forall a. Bits a => a -> Int -> a
`unsafeShiftR` Int
o))
{-# INLINE pokeWord #-}
pokeWord :: Storable a => Ptr a -> a -> IO S
pokeWord :: forall a. Storable a => Ptr a -> a -> IO S
pokeWord Ptr a
op a
w = forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr a
op a
w forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> forall (m :: * -> *) a. Monad m => Ptr a -> m S
skipByte Ptr a
op
{-# INLINE pokeWord' #-}
pokeWord' :: Storable a => Ptr a -> a -> IO (Ptr b)
pokeWord' :: forall a b. Storable a => Ptr a -> a -> IO (Ptr b)
pokeWord' Ptr a
op a
w = forall a. Storable a => Ptr a -> a -> IO ()
poke Ptr a
op a
w forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> forall (m :: * -> *) a. Monad m => a -> m a
return (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr a
op Int
1)
{-# INLINE pokeW #-}
pokeW :: Storable a => (t -> a) -> Ptr a1 -> t -> IO ()
pokeW :: forall a t a1. Storable a => (t -> a) -> Ptr a1 -> t -> IO ()
pokeW t -> a
conv Ptr a1
op t
t = forall a. Storable a => Ptr a -> a -> IO ()
poke (forall a b. Ptr a -> Ptr b
castPtr Ptr a1
op) (t -> a
conv t
t)
{-# INLINE poke64 #-}
poke64 :: (t -> Word64) -> Ptr a -> t -> IO ()
poke64 :: forall t a. (t -> Word64) -> Ptr a -> t -> IO ()
poke64 t -> Word64
conv Ptr a
op t
t = forall a. Storable a => Ptr a -> a -> IO ()
poke (forall a b. Ptr a -> Ptr b
castPtr Ptr a
op) (t -> Word64
conv t
t)
{-# INLINE skipByte #-}
skipByte :: Monad m => Ptr a -> m S
skipByte :: forall (m :: * -> *) a. Monad m => Ptr a -> m S
skipByte Ptr a
op = forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr a
op Int
1) Word8
0 Int
0)
{-# INLINE skipBytes #-}
skipBytes :: Monad m => Ptr a -> Int -> m S
skipBytes :: forall (m :: * -> *) a. Monad m => Ptr a -> Int -> m S
skipBytes Ptr a
op Int
n = forall (m :: * -> *) a. Monad m => a -> m a
return (Ptr Word8 -> Word8 -> Int -> S
S (forall a b. Ptr a -> Int -> Ptr b
plusPtr Ptr a
op Int
n) Word8
0 Int
0)
writeBS :: B.ByteString -> Ptr Word8 -> IO (Ptr Word8)
writeBS :: ByteString -> Ptr Word8 -> IO (Ptr Word8)
writeBS ByteString
bs Ptr Word8
op
| ByteString -> Int
B.length ByteString
bs forall a. Eq a => a -> a -> Bool
== Int
0 = forall (m :: * -> *) a. Monad m => a -> m a
return Ptr Word8
op
| Bool
otherwise =
let (ByteString
h, ByteString
t) = Int -> ByteString -> (ByteString, ByteString)
B.splitAt Int
255 ByteString
bs
in forall a b. Storable a => Ptr a -> a -> IO (Ptr b)
pokeWord' Ptr Word8
op (forall a b. (Integral a, Num b) => a -> b
fromIntegral forall a b. (a -> b) -> a -> b
$ ByteString -> Int
B.length ByteString
h :: Word8) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ByteString -> Ptr Word8 -> IO (Ptr Word8)
pokeByteString ByteString
h forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>=
ByteString -> Ptr Word8 -> IO (Ptr Word8)
writeBS ByteString
t
{-# INLINE asWord64 #-}
asWord64 :: Integral a => a -> Word64
asWord64 :: forall a. Integral a => a -> Word64
asWord64 = forall a b. (Integral a, Num b) => a -> b
fromIntegral
{-# INLINE asWord32 #-}
asWord32 :: Integral a => a -> Word32
asWord32 :: forall a. Integral a => a -> Word32
asWord32 = forall a b. (Integral a, Num b) => a -> b
fromIntegral
{-# INLINE asWord8 #-}
asWord8 :: Integral a => a -> Word8
asWord8 :: forall a. Integral a => a -> Word8
asWord8 = forall a b. (Integral a, Num b) => a -> b
fromIntegral