{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE Safe #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeFamilies #-}
module Text.Megaparsec.Byte.Binary
(
BinaryChunk (..),
anyLE,
anyBE,
word8,
word16le,
word16be,
word32le,
word32be,
word64le,
word64be,
int8,
int16le,
int16be,
int32le,
int32be,
int64le,
int64be,
)
where
import Data.Bits
import qualified Data.ByteString as B
import qualified Data.ByteString.Lazy as BL
import Data.Int
import Data.Word
import Text.Megaparsec
class BinaryChunk chunk where
convertChunkBE :: (Bits a, Num a) => chunk -> a
convertChunkLE :: (Bits a, Num a) => chunk -> a
instance BinaryChunk B.ByteString where
convertChunkBE :: forall a. (Bits a, Num a) => ByteString -> a
convertChunkBE = (a -> Word8 -> a) -> a -> ByteString -> a
forall a. (a -> Word8 -> a) -> a -> ByteString -> a
B.foldl' a -> Word8 -> a
forall {a} {a}. (Bits a, Integral a, Num a) => a -> a -> a
go a
0
where
go :: a -> a -> a
go a
acc a
byte = (a
acc a -> Int -> a
forall a. Bits a => a -> Int -> a
`unsafeShiftL` Int
8) a -> a -> a
forall a. Bits a => a -> a -> a
.|. a -> a
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
byte
convertChunkLE :: forall a. (Bits a, Num a) => ByteString -> a
convertChunkLE = (a -> Word8 -> a) -> a -> ByteString -> a
forall a. (a -> Word8 -> a) -> a -> ByteString -> a
B.foldl' a -> Word8 -> a
forall {a} {a}. (Bits a, Integral a, Num a) => a -> a -> a
go a
0
where
go :: a -> a -> a
go a
acc a
byte = (a
acc a -> a -> a
forall a. Bits a => a -> a -> a
.|. a -> a
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
byte) a -> Int -> a
forall a. Bits a => a -> Int -> a
`rotateR` Int
8
instance BinaryChunk BL.ByteString where
convertChunkBE :: forall a. (Bits a, Num a) => ByteString -> a
convertChunkBE = (a -> Word8 -> a) -> a -> ByteString -> a
forall a. (a -> Word8 -> a) -> a -> ByteString -> a
BL.foldl' a -> Word8 -> a
forall {a} {a}. (Bits a, Integral a, Num a) => a -> a -> a
go a
0
where
go :: a -> a -> a
go a
acc a
byte = (a
acc a -> Int -> a
forall a. Bits a => a -> Int -> a
`unsafeShiftL` Int
8) a -> a -> a
forall a. Bits a => a -> a -> a
.|. a -> a
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
byte
convertChunkLE :: forall a. (Bits a, Num a) => ByteString -> a
convertChunkLE = (a -> Word8 -> a) -> a -> ByteString -> a
forall a. (a -> Word8 -> a) -> a -> ByteString -> a
BL.foldl' a -> Word8 -> a
forall {a} {a}. (Bits a, Integral a, Num a) => a -> a -> a
go a
0
where
go :: a -> a -> a
go a
acc a
byte = (a
acc a -> a -> a
forall a. Bits a => a -> a -> a
.|. a -> a
forall a b. (Integral a, Num b) => a -> b
fromIntegral a
byte) a -> Int -> a
forall a. Bits a => a -> Int -> a
`rotateR` Int
8
anyLE ::
forall a e s m.
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String ->
m a
anyLE :: forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE Maybe String
mlabel = Tokens s -> a
forall a. (Bits a, Num a) => Tokens s -> a
forall chunk a. (BinaryChunk chunk, Bits a, Num a) => chunk -> a
convertChunkLE (Tokens s -> a) -> m (Tokens s) -> m a
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe String -> Int -> m (Tokens s)
forall e s (m :: * -> *).
MonadParsec e s m =>
Maybe String -> Int -> m (Tokens s)
takeP Maybe String
mlabel (forall a. FiniteBits a => Int
finiteByteSize @a)
{-# INLINE anyLE #-}
anyBE ::
forall a e s m.
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String ->
m a
anyBE :: forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE Maybe String
mlabel = Tokens s -> a
forall a. (Bits a, Num a) => Tokens s -> a
forall chunk a. (BinaryChunk chunk, Bits a, Num a) => chunk -> a
convertChunkBE (Tokens s -> a) -> m (Tokens s) -> m a
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe String -> Int -> m (Tokens s)
forall e s (m :: * -> *).
MonadParsec e s m =>
Maybe String -> Int -> m (Tokens s)
takeP Maybe String
mlabel (forall a. FiniteBits a => Int
finiteByteSize @a)
{-# INLINE anyBE #-}
word8 :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word8
word8 :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word8
word8 = Maybe String -> m Word8
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"8 bit word")
{-# INLINE word8 #-}
word16le :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word16
word16le :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word16
word16le = Maybe String -> m Word16
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE (String -> Maybe String
forall a. a -> Maybe a
Just String
"little-endian 16 bit word")
{-# INLINE word16le #-}
word16be :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word16
word16be :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word16
word16be = Maybe String -> m Word16
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"big-endian 16 bit word")
{-# INLINE word16be #-}
word32le :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word32
word32le :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word32
word32le = Maybe String -> m Word32
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE (String -> Maybe String
forall a. a -> Maybe a
Just String
"little-endian 32 bit word")
{-# INLINE word32le #-}
word32be :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word32
word32be :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word32
word32be = Maybe String -> m Word32
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"big-endian 32 bit word")
{-# INLINE word32be #-}
word64le :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word64
word64le :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word64
word64le = Maybe String -> m Word64
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE (String -> Maybe String
forall a. a -> Maybe a
Just String
"little-endian 64 word")
{-# INLINE word64le #-}
word64be :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Word64
word64be :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Word64
word64be = Maybe String -> m Word64
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"big-endian 64 word")
{-# INLINE word64be #-}
int8 :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int8
int8 :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int8
int8 = Maybe String -> m Int8
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"8 bit int")
{-# INLINE int8 #-}
int16le :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int16
int16le :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int16
int16le = Maybe String -> m Int16
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE (String -> Maybe String
forall a. a -> Maybe a
Just String
"little-endian 16 bit int")
{-# INLINE int16le #-}
int16be :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int16
int16be :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int16
int16be = Maybe String -> m Int16
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"big-endian 16 bit int")
{-# INLINE int16be #-}
int32le :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int32
int32le :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int32
int32le = Maybe String -> m Int32
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE (String -> Maybe String
forall a. a -> Maybe a
Just String
"little-endian 32 bit int")
{-# INLINE int32le #-}
int32be :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int32
int32be :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int32
int32be = Maybe String -> m Int32
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"big-endian 32 bit int")
{-# INLINE int32be #-}
int64le :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int64
int64le :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int64
int64le = Maybe String -> m Int64
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyLE (String -> Maybe String
forall a. a -> Maybe a
Just String
"little-endian 64 int")
{-# INLINE int64le #-}
int64be :: (MonadParsec e s m, BinaryChunk (Tokens s)) => m Int64
int64be :: forall e s (m :: * -> *).
(MonadParsec e s m, BinaryChunk (Tokens s)) =>
m Int64
int64be = Maybe String -> m Int64
forall a e s (m :: * -> *).
(MonadParsec e s m, FiniteBits a, Num a, BinaryChunk (Tokens s)) =>
Maybe String -> m a
anyBE (String -> Maybe String
forall a. a -> Maybe a
Just String
"big-endian 64 int")
{-# INLINE int64be #-}
finiteByteSize :: forall a. (FiniteBits a) => Int
finiteByteSize :: forall a. FiniteBits a => Int
finiteByteSize = forall b. FiniteBits b => b -> Int
finiteBitSize @a a
forall a. HasCallStack => a
undefined Int -> Int -> Int
forall {a}. Integral a => a -> a -> a
`ceilDiv` Int
8
where
ceilDiv :: a -> a -> a
ceilDiv a
x a
y = (a
x a -> a -> a
forall a. Num a => a -> a -> a
+ a
y a -> a -> a
forall a. Num a => a -> a -> a
- a
1) a -> a -> a
forall {a}. Integral a => a -> a -> a
`div` a
y
{-# INLINE finiteByteSize #-}