-- | Machine integer parsers.

module FlatParse.Stateful.Integers
  (
  -- * Native byte order
    anyWord8, anyWord16, anyWord32, anyWord64
  , anyInt8,  anyInt16,  anyInt32,  anyInt64
  , anyWord, anyInt

  -- * Explicit endianness
  -- $explicit-endianness
  , anyWord16le, anyWord16be
  , anyWord32le, anyWord32be
  , anyWord64le, anyWord64be
  , anyInt16le,  anyInt16be
  , anyInt32le,  anyInt32be
  , anyInt64le,  anyInt64be

  -- * Value assertions
  , word8

  -- * CPS parsers
  , withAnyWord8, withAnyWord16, withAnyWord32, withAnyWord64
  , withAnyInt8,  withAnyInt16,  withAnyInt32,  withAnyInt64
  , withAnyWord, withAnyInt

  -- * Unsafe
  -- $unsafe
  , anyWord8Unsafe

  -- ** Value assertions
  , word8Unsafe, word16Unsafe, word32Unsafe, word64Unsafe

  -- * Helper definitions
  , withAnySized#, withAnySizedUnsafe#
  , sizedUnsafe#
  ) where

import FlatParse.Stateful.Parser
import FlatParse.Stateful.Base ( withEnsure# )

import FlatParse.Common.Assorted ( word16ToInt16, word32ToInt32, word64ToInt64 )

import FlatParse.Common.GHCExts
import GHC.Word
import GHC.Int

import Control.Applicative ( Alternative(empty) )

--------------------------------------------------------------------------------

-- | Helper for defining CPS parsers for types of a constant byte size (i.e.
--   machine integers).
--
-- Call this with an @indexXYZOffAddr@ primop (e.g.
-- 'GHC.Exts.indexWord8OffAddr') and the size in bytes of the type you're
-- parsing.
withAnySized#
    :: Int# -> (Addr# -> Int# -> a) -> (a -> ParserT st r e ret)
    -> ParserT st r e ret
withAnySized# :: forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
size# Addr# -> Int# -> a
indexOffAddr a -> ParserT st r e ret
p =
    forall (st :: ZeroBitType) r e ret.
Int# -> ParserT st r e ret -> ParserT st r e ret
withEnsure# Int#
size# (forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySizedUnsafe# Int#
size# Addr# -> Int# -> a
indexOffAddr a -> ParserT st r e ret
p)
{-# inline withAnySized# #-}

-- | Unsafe helper for defining CPS parsers for types of a constant byte size
--   (i.e. machine integers).
--
-- Is really just syntactic sugar for applying the given parser and shifting the
-- buffer along.
--
-- The caller must guarantee that the input has enough bytes.
withAnySizedUnsafe#
    :: Int# -> (Addr# -> Int# -> a) -> (a -> ParserT st r e ret)
    -> ParserT st r e ret
withAnySizedUnsafe# :: forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySizedUnsafe# Int#
size# Addr# -> Int# -> a
indexOffAddr a -> ParserT st r e ret
p = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT \ForeignPtrContents
fp !r
r Addr#
eob Addr#
buf Int#
n st
st ->
  let a :: a
a    = Addr# -> Int# -> a
indexOffAddr Addr#
buf Int#
0#
      buf' :: Addr#
buf' = Addr# -> Int# -> Addr#
plusAddr# Addr#
buf Int#
size#
  in  forall (st :: ZeroBitType) r e a.
ParserT st r e a
-> ForeignPtrContents
-> r
-> Addr#
-> Addr#
-> Int#
-> st
-> Res# st e a
runParserT# (a -> ParserT st r e ret
p a
a) ForeignPtrContents
fp r
r Addr#
eob Addr#
buf' Int#
n st
st
{-# inline withAnySizedUnsafe# #-}

-- | Parse any 'Word8' (CPS).
withAnyWord8 :: (Word8 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord8 :: forall (st :: ZeroBitType) r e ret.
(Word8 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord8 Word8 -> ParserT st r e ret
p = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT \ForeignPtrContents
fp !r
r Addr#
eob Addr#
buf Int#
n st
st -> case Addr# -> Addr# -> Int#
eqAddr# Addr#
eob Addr#
buf of
  Int#
1# -> forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
  Int#
_  -> let w# :: Word8#
w# = Addr# -> Int# -> Word8#
indexWord8OffAddr# Addr#
buf Int#
0#
        in  forall (st :: ZeroBitType) r e a.
ParserT st r e a
-> ForeignPtrContents
-> r
-> Addr#
-> Addr#
-> Int#
-> st
-> Res# st e a
runParserT# (Word8 -> ParserT st r e ret
p (Word8# -> Word8
W8# Word8#
w#)) ForeignPtrContents
fp r
r Addr#
eob (Addr# -> Int# -> Addr#
plusAddr# Addr#
buf Int#
1#) Int#
n st
st
{-# inline withAnyWord8 #-}

-- | Parse any 'Word16' (native byte order) (CPS).
withAnyWord16 :: (Word16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord16 :: forall (st :: ZeroBitType) r e ret.
(Word16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord16 = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
2# (\Addr#
a Int#
i -> Word16# -> Word16
W16# (Addr# -> Int# -> Word16#
indexWord16OffAddr# Addr#
a Int#
i))
{-# inline withAnyWord16 #-}

-- | Parse any 'Word32' (native byte order) (CPS).
withAnyWord32 :: (Word32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord32 :: forall (st :: ZeroBitType) r e ret.
(Word32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord32 = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
4# (\Addr#
a Int#
i -> Word32# -> Word32
W32# (Addr# -> Int# -> Word32#
indexWord32OffAddr# Addr#
a Int#
i))
{-# inline withAnyWord32 #-}

-- | Parse any 'Word64' (native byte order) (CPS).
withAnyWord64 :: (Word64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord64 :: forall (st :: ZeroBitType) r e ret.
(Word64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord64 = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
8# (\Addr#
a Int#
i -> Word# -> Word64
W64# (Addr# -> Int# -> Word#
indexWord64OffAddr# Addr#
a Int#
i))
{-# inline withAnyWord64 #-}

-- | Parse any 'Int8' (CPS).
withAnyInt8 :: (Int8 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt8 :: forall (st :: ZeroBitType) r e ret.
(Int8 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt8 Int8 -> ParserT st r e ret
p = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT \ForeignPtrContents
fp !r
r Addr#
eob Addr#
buf Int#
n st
st -> case Addr# -> Addr# -> Int#
eqAddr# Addr#
eob Addr#
buf of
  Int#
1# -> forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
  Int#
_  -> let i# :: Int8#
i# = Addr# -> Int# -> Int8#
indexInt8OffAddr# Addr#
buf Int#
0#
        in  forall (st :: ZeroBitType) r e a.
ParserT st r e a
-> ForeignPtrContents
-> r
-> Addr#
-> Addr#
-> Int#
-> st
-> Res# st e a
runParserT# (Int8 -> ParserT st r e ret
p (Int8# -> Int8
I8# Int8#
i#)) ForeignPtrContents
fp r
r Addr#
eob (Addr# -> Int# -> Addr#
plusAddr# Addr#
buf Int#
1#) Int#
n st
st
{-# inline withAnyInt8 #-}

-- | Parse any 'Int16' (native byte order) (CPS).
withAnyInt16 :: (Int16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt16 :: forall (st :: ZeroBitType) r e ret.
(Int16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt16 = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
2# (\Addr#
a Int#
i -> Int16# -> Int16
I16# (Addr# -> Int# -> Int16#
indexInt16OffAddr# Addr#
a Int#
i))
{-# inline withAnyInt16 #-}

-- | Parse any 'Int32' (native byte order) (CPS).
withAnyInt32 :: (Int32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt32 :: forall (st :: ZeroBitType) r e ret.
(Int32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt32 = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
4# (\Addr#
a Int#
i -> Int32# -> Int32
I32# (Addr# -> Int# -> Int32#
indexInt32OffAddr# Addr#
a Int#
i))
{-# inline withAnyInt32 #-}

-- | Parse any 'Int64' (native byte order) (CPS).
withAnyInt64 :: (Int64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt64 :: forall (st :: ZeroBitType) r e ret.
(Int64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt64 = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySized# Int#
8# (\Addr#
a Int#
i -> Int# -> Int64
I64# (Addr# -> Int# -> Int#
indexInt64OffAddr# Addr#
a Int#
i))
{-# inline withAnyInt64 #-}

-- TODO assumes 64-bit platform
-- | Parse any 'Word' (native size) (CPS).
withAnyWord :: (Word -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord :: forall (st :: ZeroBitType) r e ret.
(Word -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord Word -> ParserT st r e ret
p = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT \ForeignPtrContents
fp !r
r Addr#
eob Addr#
buf Int#
n st
st -> case Int#
8# Int# -> Int# -> Int#
<=# Addr# -> Addr# -> Int#
minusAddr# Addr#
eob Addr#
buf of
  Int#
0# -> forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
  Int#
_  -> let w# :: Word#
w# = Addr# -> Int# -> Word#
indexWordOffAddr# Addr#
buf Int#
0#
        in  forall (st :: ZeroBitType) r e a.
ParserT st r e a
-> ForeignPtrContents
-> r
-> Addr#
-> Addr#
-> Int#
-> st
-> Res# st e a
runParserT# (Word -> ParserT st r e ret
p (Word# -> Word
W# Word#
w#)) ForeignPtrContents
fp r
r Addr#
eob (Addr# -> Int# -> Addr#
plusAddr# Addr#
buf Int#
8#) Int#
n st
st
{-# inline withAnyWord #-}

-- TODO assumes 64-bit platform
-- | Parse any 'Int' (native size) (CPS).
withAnyInt :: (Int -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt :: forall (st :: ZeroBitType) r e ret.
(Int -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt Int -> ParserT st r e ret
p = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT \ForeignPtrContents
fp !r
r Addr#
eob Addr#
buf Int#
n st
st -> case Int#
8# Int# -> Int# -> Int#
<=# Addr# -> Addr# -> Int#
minusAddr# Addr#
eob Addr#
buf of
  Int#
0# -> forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
  Int#
_  -> let i# :: Int#
i# = Addr# -> Int# -> Int#
indexIntOffAddr# Addr#
buf Int#
0#
        in  forall (st :: ZeroBitType) r e a.
ParserT st r e a
-> ForeignPtrContents
-> r
-> Addr#
-> Addr#
-> Int#
-> st
-> Res# st e a
runParserT# (Int -> ParserT st r e ret
p (Int# -> Int
I# Int#
i#)) ForeignPtrContents
fp r
r Addr#
eob (Addr# -> Int# -> Addr#
plusAddr# Addr#
buf Int#
8#) Int#
n st
st
{-# inline withAnyInt #-}

--------------------------------------------------------------------------------

-- | Parse any 'Word8'.
anyWord8 :: ParserT st r e Word8
anyWord8 :: forall (st :: ZeroBitType) r e. ParserT st r e Word8
anyWord8 = forall (st :: ZeroBitType) r e ret.
(Word8 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord8 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyWord8 #-}

-- | Parse any 'Word16' (native byte order).
anyWord16 :: ParserT st r e Word16
anyWord16 :: forall (st :: ZeroBitType) r e. ParserT st r e Word16
anyWord16 = forall (st :: ZeroBitType) r e ret.
(Word16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord16 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyWord16 #-}

-- | Parse any 'Word32' (native byte order).
anyWord32 :: ParserT st r e Word32
anyWord32 :: forall (st :: ZeroBitType) r e. ParserT st r e Word32
anyWord32 = forall (st :: ZeroBitType) r e ret.
(Word32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord32 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyWord32 #-}

-- | Parse any 'Word64' (native byte order).
anyWord64 :: ParserT st r e Word64
anyWord64 :: forall (st :: ZeroBitType) r e. ParserT st r e Word64
anyWord64 = forall (st :: ZeroBitType) r e ret.
(Word64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord64 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyWord64 #-}

-- | Parse any 'Int8'.
anyInt8 :: ParserT st r e Int8
anyInt8 :: forall (st :: ZeroBitType) r e. ParserT st r e Int8
anyInt8 = forall (st :: ZeroBitType) r e ret.
(Int8 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt8 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyInt8 #-}

-- | Parse any 'Int16' (native byte order).
anyInt16 :: ParserT st r e Int16
anyInt16 :: forall (st :: ZeroBitType) r e. ParserT st r e Int16
anyInt16 = forall (st :: ZeroBitType) r e ret.
(Int16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt16 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyInt16 #-}

-- | Parse any 'Int32' (native byte order).
anyInt32 :: ParserT st r e Int32
anyInt32 :: forall (st :: ZeroBitType) r e. ParserT st r e Int32
anyInt32 = forall (st :: ZeroBitType) r e ret.
(Int32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt32 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyInt32 #-}

-- | Parse any 'Int64' (native byte order).
anyInt64 :: ParserT st r e Int64
anyInt64 :: forall (st :: ZeroBitType) r e. ParserT st r e Int64
anyInt64 = forall (st :: ZeroBitType) r e ret.
(Int64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt64 forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyInt64 #-}

-- TODO 'withAnyWord' assumes 64-bit platform
-- | Parse any 'Word' (native size).
anyWord :: ParserT st r e Word
anyWord :: forall (st :: ZeroBitType) r e. ParserT st r e Word
anyWord = forall (st :: ZeroBitType) r e ret.
(Word -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyWord #-}

-- TODO 'withAnyInt' assumes 64-bit platform
-- | Parse any 'Int' (native size).
anyInt :: ParserT st r e Int
anyInt :: forall (st :: ZeroBitType) r e. ParserT st r e Int
anyInt = forall (st :: ZeroBitType) r e ret.
(Int -> ParserT st r e ret) -> ParserT st r e ret
withAnyInt forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyInt #-}

--------------------------------------------------------------------------------

{- $explicit-endianness
Native endianness parsers are used where possible. For non-native endianness
parsers, we parse then use the corresponding @byteSwapX@ function. On x86, this
is inlined as a single @BSWAP@ instruction.
-}

-- | Parse any 'Word16' (little-endian).
anyWord16le :: ParserT st r e Word16
#ifdef WORDS_BIGENDIAN
anyWord16le = withAnyWord16 (pure . byteSwap16)
#else
anyWord16le :: forall (st :: ZeroBitType) r e. ParserT st r e Word16
anyWord16le = forall (st :: ZeroBitType) r e. ParserT st r e Word16
anyWord16
#endif
{-# inline anyWord16le #-}

-- | Parse any 'Word16' (big-endian).
anyWord16be :: ParserT st r e Word16
#ifdef WORDS_BIGENDIAN
anyWord16be = anyWord16
#else
anyWord16be :: forall (st :: ZeroBitType) r e. ParserT st r e Word16
anyWord16be = forall (st :: ZeroBitType) r e ret.
(Word16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord16 (forall (f :: * -> *) a. Applicative f => a -> f a
pure forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word16 -> Word16
byteSwap16)
#endif
{-# inline anyWord16be #-}

-- | Parse any 'Word32' (little-endian).
anyWord32le :: ParserT st r e Word32
#ifdef WORDS_BIGENDIAN
anyWord32le = withAnyWord32 (pure . byteSwap32)
#else
anyWord32le :: forall (st :: ZeroBitType) r e. ParserT st r e Word32
anyWord32le = forall (st :: ZeroBitType) r e. ParserT st r e Word32
anyWord32
#endif
{-# inline anyWord32le #-}

-- | Parse any 'Word32' (big-endian).
anyWord32be :: ParserT st r e Word32
#ifdef WORDS_BIGENDIAN
anyWord32be = anyWord32
#else
anyWord32be :: forall (st :: ZeroBitType) r e. ParserT st r e Word32
anyWord32be = forall (st :: ZeroBitType) r e ret.
(Word32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord32 (forall (f :: * -> *) a. Applicative f => a -> f a
pure forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word32 -> Word32
byteSwap32)
#endif
{-# inline anyWord32be #-}

-- | Parse any 'Word64' (little-endian).
anyWord64le :: ParserT st r e Word64
#ifdef WORDS_BIGENDIAN
anyWord64le = withAnyWord64 (pure . byteSwap64)
#else
anyWord64le :: forall (st :: ZeroBitType) r e. ParserT st r e Word64
anyWord64le = forall (st :: ZeroBitType) r e. ParserT st r e Word64
anyWord64
#endif
{-# inline anyWord64le #-}

-- | Parse any 'Word64' (big-endian).
anyWord64be :: ParserT st r e Word64
#ifdef WORDS_BIGENDIAN
anyWord64be = anyWord64
#else
anyWord64be :: forall (st :: ZeroBitType) r e. ParserT st r e Word64
anyWord64be = forall (st :: ZeroBitType) r e ret.
(Word64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord64 (forall (f :: * -> *) a. Applicative f => a -> f a
pure forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word64 -> Word64
byteSwap64)
#endif
{-# inline anyWord64be #-}

-- | Parse any 'Int16' (little-endian).
anyInt16le :: ParserT st r e Int16
#ifdef WORDS_BIGENDIAN
anyInt16le = withAnyWord16 (pure . word16ToInt16 . byteSwap16)
#else
anyInt16le :: forall (st :: ZeroBitType) r e. ParserT st r e Int16
anyInt16le = forall (st :: ZeroBitType) r e. ParserT st r e Int16
anyInt16
#endif
{-# inline anyInt16le #-}

-- | Parse any 'Int16' (big-endian).
anyInt16be :: ParserT st r e Int16
#ifdef WORDS_BIGENDIAN
anyInt16be = anyInt16
#else
anyInt16be :: forall (st :: ZeroBitType) r e. ParserT st r e Int16
anyInt16be = forall (st :: ZeroBitType) r e ret.
(Word16 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord16 (forall (f :: * -> *) a. Applicative f => a -> f a
pure forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word16 -> Int16
word16ToInt16 forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word16 -> Word16
byteSwap16)
#endif
{-# inline anyInt16be #-}

-- | Parse any 'Int32' (little-endian).
anyInt32le :: ParserT st r e Int32
#ifdef WORDS_BIGENDIAN
anyInt32le = withAnyWord32 (pure . word32ToInt32 . byteSwap32)
#else
anyInt32le :: forall (st :: ZeroBitType) r e. ParserT st r e Int32
anyInt32le = forall (st :: ZeroBitType) r e. ParserT st r e Int32
anyInt32
#endif
{-# inline anyInt32le #-}

-- | Parse any 'Int32' (big-endian).
anyInt32be :: ParserT st r e Int32
#ifdef WORDS_BIGENDIAN
anyInt32be = anyInt32
#else
anyInt32be :: forall (st :: ZeroBitType) r e. ParserT st r e Int32
anyInt32be = forall (st :: ZeroBitType) r e ret.
(Word32 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord32 (forall (f :: * -> *) a. Applicative f => a -> f a
pure forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word32 -> Int32
word32ToInt32 forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word32 -> Word32
byteSwap32)
#endif
{-# inline anyInt32be #-}

-- | Parse any 'Int64' (little-endian).
anyInt64le :: ParserT st r e Int64
#ifdef WORDS_BIGENDIAN
anyInt64le = withAnyWord64 (pure . word64ToInt64 . byteSwap64)
#else
anyInt64le :: forall (st :: ZeroBitType) r e. ParserT st r e Int64
anyInt64le = forall (st :: ZeroBitType) r e. ParserT st r e Int64
anyInt64
#endif
{-# inline anyInt64le #-}

-- | Parse any 'Int64' (big-endian).
anyInt64be :: ParserT st r e Int64
#ifdef WORDS_BIGENDIAN
anyInt64be = anyInt64
#else
anyInt64be :: forall (st :: ZeroBitType) r e. ParserT st r e Int64
anyInt64be = forall (st :: ZeroBitType) r e ret.
(Word64 -> ParserT st r e ret) -> ParserT st r e ret
withAnyWord64 (forall (f :: * -> *) a. Applicative f => a -> f a
pure forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word64 -> Int64
word64ToInt64 forall b c a. (b -> c) -> (a -> b) -> a -> c
. Word64 -> Word64
byteSwap64)
#endif
{-# inline anyInt64be #-}

--------------------------------------------------------------------------------

-- | Read the next 1 byte and assert its value as a 'Word8'.
word8 :: Word8 -> ParserT st r e ()
word8 :: forall (st :: ZeroBitType) r e. Word8 -> ParserT st r e ()
word8 Word8
wExpected = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT \ForeignPtrContents
fp !r
r Addr#
eob Addr#
buf Int#
n st
st -> case Addr# -> Addr# -> Int#
eqAddr# Addr#
eob Addr#
buf of
  Int#
1# -> forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
  Int#
_  -> let w# :: Word8#
w# = Addr# -> Int# -> Word8#
indexWord8OffAddr# Addr#
buf Int#
0#
        in  if   Word8# -> Word8
W8# Word8#
w# forall a. Eq a => a -> a -> Bool
== Word8
wExpected
            then forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st () (Addr# -> Int# -> Addr#
plusAddr# Addr#
buf Int#
1#) Int#
n
            else forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
{-# inline word8 #-}

--------------------------------------------------------------------------------

{- $unsafe
These unsafe parsers and helpers may be useful for efficient parsing in special
situations e.g. you already know that the input has enough bytes. You should
only use them if you can assert their necessary guarantees (see the individual
function documentation).
-}

-- | Unsafe helper for defining parsers for types of a constant byte size (i.e.
--   machine integers) which assert the parsed value's... value.
--
-- Call this with an @indexXYZOffAddr@ primop (e.g.
-- 'GHC.Exts.indexWord8OffAddr'), the size in bytes of the type you're parsing,
-- and the expected value to test the parsed value against.
--
-- The caller must guarantee that the input has enough bytes.
sizedUnsafe# :: Eq a => Int# -> (Addr# -> Int# -> a) -> a -> ParserT st r e ()
sizedUnsafe# :: forall a (st :: ZeroBitType) r e.
Eq a =>
Int# -> (Addr# -> Int# -> a) -> a -> ParserT st r e ()
sizedUnsafe# Int#
size# Addr# -> Int# -> a
indexOffAddr a
aExpected =
    forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySizedUnsafe# Int#
size# Addr# -> Int# -> a
indexOffAddr forall {f :: * -> *}. Alternative f => a -> f ()
go
  where
    go :: a -> f ()
go a
aParsed =
        if   a
aParsed forall a. Eq a => a -> a -> Bool
== a
aExpected
        then forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
        else forall (f :: * -> *) a. Alternative f => f a
empty
{-# inline sizedUnsafe# #-}

-- | Unsafely read the next 1 byte and assert its value as a 'Word8'.
--
-- The caller must guarantee that the input has enough bytes.
word8Unsafe :: Word8 -> ParserT st r e ()
word8Unsafe :: forall (st :: ZeroBitType) r e. Word8 -> ParserT st r e ()
word8Unsafe = forall a (st :: ZeroBitType) r e.
Eq a =>
Int# -> (Addr# -> Int# -> a) -> a -> ParserT st r e ()
sizedUnsafe# Int#
1# (\Addr#
a Int#
i -> Word8# -> Word8
W8# (Addr# -> Int# -> Word8#
indexWord8OffAddr# Addr#
a Int#
i))
{-# inline word8Unsafe #-}

-- | Unsafely read the next 2 bytes and assert their value as a 'Word16'
--   (native byte order).
--
-- The caller must guarantee that the input has enough bytes.
word16Unsafe :: Word16 -> ParserT st r e ()
word16Unsafe :: forall (st :: ZeroBitType) r e. Word16 -> ParserT st r e ()
word16Unsafe = forall a (st :: ZeroBitType) r e.
Eq a =>
Int# -> (Addr# -> Int# -> a) -> a -> ParserT st r e ()
sizedUnsafe# Int#
2# (\Addr#
a Int#
i -> Word16# -> Word16
W16# (Addr# -> Int# -> Word16#
indexWord16OffAddr# Addr#
a Int#
i))
{-# inline word16Unsafe #-}

-- | Unsafely read the next 4 bytes and assert their value as a 'Word32'.
--   (native byte order).
--
-- The caller must guarantee that the input has enough bytes.
word32Unsafe :: Word32 -> ParserT st r e ()
word32Unsafe :: forall (st :: ZeroBitType) r e. Word32 -> ParserT st r e ()
word32Unsafe = forall a (st :: ZeroBitType) r e.
Eq a =>
Int# -> (Addr# -> Int# -> a) -> a -> ParserT st r e ()
sizedUnsafe# Int#
4# (\Addr#
a Int#
i -> Word32# -> Word32
W32# (Addr# -> Int# -> Word32#
indexWord32OffAddr# Addr#
a Int#
i))
{-# inline word32Unsafe #-}

-- | Unsafely read the next 8 bytes and assert their value as a 'Word64'.
--   (native byte order).
--
-- The caller must guarantee that the input has enough bytes.
word64Unsafe :: Word64 -> ParserT st r e ()
word64Unsafe :: forall (st :: ZeroBitType) r e. Word64 -> ParserT st r e ()
word64Unsafe = forall a (st :: ZeroBitType) r e.
Eq a =>
Int# -> (Addr# -> Int# -> a) -> a -> ParserT st r e ()
sizedUnsafe# Int#
8# (\Addr#
a Int#
i -> Word# -> Word64
W64# (Addr# -> Int# -> Word#
indexWord64OffAddr# Addr#
a Int#
i))
{-# inline word64Unsafe #-}

--------------------------------------------------------------------------------

-- | Unsafely parse any 'Word8', without asserting the input is non-empty.
--
-- The caller must guarantee that the input has enough bytes.
anyWord8Unsafe :: ParserT st r e Word8
anyWord8Unsafe :: forall (st :: ZeroBitType) r e. ParserT st r e Word8
anyWord8Unsafe = forall a (st :: ZeroBitType) r e ret.
Int#
-> (Addr# -> Int# -> a)
-> (a -> ParserT st r e ret)
-> ParserT st r e ret
withAnySizedUnsafe# Int#
1# (\Addr#
a Int#
i -> Word8# -> Word8
W8# (Addr# -> Int# -> Word8#
indexWord8OffAddr# Addr#
a Int#
i)) forall (f :: * -> *) a. Applicative f => a -> f a
pure
{-# inline anyWord8Unsafe #-}