-- | Minimal parser definition.

{-# LANGUAGE UnboxedTuples #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE DataKinds #-} -- needed for manual ZeroBitType def (unsure why)
{-# LANGUAGE FlexibleInstances #-}

module FlatParse.Stateful.Parser
  (
  -- * Parser
    ParserT(..)
  , Parser, ParserIO, ParserST

  -- ** Result
  , type Res#
  , pattern OK#, pattern Err#, pattern Fail#

  -- *** Internal
  , type ResI#

  -- * Choice operator (defined with right associativity)
  , (<|>)
  ) where

import FlatParse.Common.GHCExts ( Addr#, unsafeCoerce#, ZeroBitType, Int# )
import FlatParse.Common.Parser

import GHC.ForeignPtr ( ForeignPtrContents )

import qualified Control.Applicative
import Control.Monad ( MonadPlus(..) )
import Control.Monad.IO.Class ( MonadIO(..) )
import GHC.IO ( IO(IO) )

-- | @ParserT st r e a@ is a parser with a state token type @st@, a reader
--   environment @r@, an error type @e@ and a return type @a@. The different
--   state token types support different embedded effects; see `Parser`,
--   `ParserIO` and `ParserST` below.
newtype ParserT (st :: ZeroBitType) r e a =
    ParserT { forall (st :: ZeroBitType) r e a.
ParserT st r e a
-> ForeignPtrContents
-> r
-> Addr#
-> Addr#
-> Int#
-> st
-> Res# st e a
runParserT# :: ForeignPtrContents -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a }

-- | The type of pure parsers.
type Parser     = ParserT PureMode

-- | The type of parsers which can embed `IO` actions.
type ParserIO   = ParserT IOMode

-- | The type of parsers which can embed `ST` actions.
type ParserST s = ParserT (STMode s)

-- | You may lift IO actions into a 'ParserIO' using `liftIO`.
instance MonadIO (ParserT IOMode r e) where
  liftIO :: forall a. IO a -> ParserT IOMode r e a
liftIO (IO IOMode -> (# IOMode, a #)
a) = 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#
s Int#
n IOMode
rw ->
    case IOMode -> (# IOMode, a #)
a IOMode
rw of (# IOMode
rw', a
a #) -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# IOMode
rw' a
a Addr#
s Int#
n
  {-# inline liftIO #-}

instance Functor (ParserT st r e) where
  fmap :: forall a b. (a -> b) -> ParserT st r e a -> ParserT st r e b
fmap a -> b
f (ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
g) = 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#
s Int#
n st
st -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
g ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    OK# st
st' a
a Addr#
s Int#
n -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st' (a -> b
f forall a b. (a -> b) -> a -> b
$! a
a) Addr#
s Int#
n
    Res# st e a
x             -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e a
x
  {-# inline fmap #-}

  <$ :: forall a b. a -> ParserT st r e b -> ParserT st r e a
(<$) a
a' (ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e b
g) = 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#
s Int#
n st
st -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e b
g ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    OK# st
st' b
_a Addr#
s Int#
n -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st' a
a' Addr#
s Int#
n
    Res# st e b
x              -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e b
x
  {-# inline (<$) #-}

instance Applicative (ParserT st r e) where
  pure :: forall a. a -> ParserT st r e a
pure a
a = 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#
s Int#
n st
st -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st a
a Addr#
s Int#
n
  {-# inline pure #-}
  ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e (a -> b)
ff <*> :: forall a b.
ParserT st r e (a -> b) -> ParserT st r e a -> ParserT st r e b
<*> ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa = 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#
s Int#
n st
st -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e (a -> b)
ff ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    OK# st
st' a -> b
f Addr#
s Int#
n -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st' of
      OK# st
st'' a
a Addr#
s Int#
n -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st'' (a -> b
f forall a b. (a -> b) -> a -> b
$! a
a) Addr#
s Int#
n
      Res# st e a
x              -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e a
x
    Res# st e (a -> b)
x             -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e (a -> b)
x
  {-# inline (<*>) #-}
  ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa <* :: forall a b.
ParserT st r e a -> ParserT st r e b -> ParserT st r e a
<* ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e b
fb = 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#
s Int#
n st
st -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    OK# st
st' a
a Addr#
s Int#
n -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e b
fb ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st' of
      OK# st
st'' b
_b Addr#
s Int#
n -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st'' a
a Addr#
s Int#
n
      Res# st e b
x               -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e b
x
    Res# st e a
x             -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e a
x
  {-# inline (<*) #-}
  ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa *> :: forall a b.
ParserT st r e a -> ParserT st r e b -> ParserT st r e b
*> ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e b
fb = 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#
s Int#
n st
st -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    OK# st
st' a
_a Addr#
s Int#
n -> ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e b
fb ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st'
    Res# st e a
x              -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e a
x
  {-# inline (*>) #-}

instance Monad (ParserT st r e) where
  return :: forall a. a -> ParserT st r e a
return = forall (f :: * -> *) a. Applicative f => a -> f a
pure
  {-# inline return #-}
  ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa >>= :: forall a b.
ParserT st r e a -> (a -> ParserT st r e b) -> ParserT st r e b
>>= a -> ParserT st r e b
f = 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#
s Int#
n st
st -> case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
fa ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    OK# st
st' a
a Addr#
s Int#
n -> 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 b
f a
a) ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st'
    Res# st e a
x             -> unsafeCoerce# :: forall a b. a -> b
unsafeCoerce# Res# st e a
x
  {-# inline (>>=) #-}
  >> :: forall a b.
ParserT st r e a -> ParserT st r e b -> ParserT st r e b
(>>) = forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
(*>)
  {-# inline (>>) #-}

instance Control.Applicative.Alternative (ParserT st r e) where
  empty :: forall a. ParserT st r e a
empty = 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#
s Int#
n st
st -> forall (st :: ZeroBitType) e a. st -> Res# st e a
Fail# st
st
  {-# inline empty #-}

  <|> :: forall a. ParserT st r e a -> ParserT st r e a -> ParserT st r e a
(<|>) = forall (st :: ZeroBitType) r e a.
ParserT st r e a -> ParserT st r e a -> ParserT st r e a
(<|>)
  {-# inline (Control.Applicative.<|>) #-}

  many :: forall a. ParserT st r e a -> ParserT st r e [a]
many (ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
f) = forall (st :: ZeroBitType) r e a.
(ForeignPtrContents
 -> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a)
-> ParserT st r e a
ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e [a]
go where
    go :: ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e [a]
go ForeignPtrContents
fp !r
r Addr#
eob Addr#
s Int#
n st
st =
        case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
f ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
          OK# st
st a
a Addr#
s Int#
n ->
            case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e [a]
go ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
              OK# st
st [a]
as Addr#
s Int#
n -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st (a
aforall a. a -> [a] -> [a]
:[a]
as) Addr#
s Int#
n
              Res# st e [a]
x             -> Res# st e [a]
x
          Fail# st
st'    -> forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
OK# st
st [] Addr#
s Int#
n
          Err# st
st' e
e   -> forall (st :: ZeroBitType) e a. st -> e -> Res# st e a
Err# st
st e
e
  {-# inline many #-}

  some :: forall a. ParserT st r e a -> ParserT st r e [a]
some ParserT st r e a
p = (:) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ParserT st r e a
p forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> forall (f :: * -> *) a. Alternative f => f a -> f [a]
Control.Applicative.many ParserT st r e a
p
  {-# inline some #-}

infixr 6 <|>
-- | Choose between two parsers. If the first parser fails, try the second one,
--   but if the first one throws an error, propagate the error. This operation
--   can arbitrarily backtrack.
--
-- Note: this exported operator has different fixity than the same operator in
-- `Control.Applicative`. Hide this operator if you want to use the
-- `Alternative` version.
(<|>) :: ParserT st r e a -> ParserT st r e a -> ParserT st r e a
<|> :: forall (st :: ZeroBitType) r e a.
ParserT st r e a -> ParserT st r e a -> ParserT st r e a
(<|>) (ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
f) (ParserT ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
g) = 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#
s Int#
n st
st ->
  case ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
f ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st of
    Fail# st
st' -> ForeignPtrContents
-> r -> Addr# -> Addr# -> Int# -> st -> Res# st e a
g ForeignPtrContents
fp r
r Addr#
eob Addr#
s Int#
n st
st'
    Res# st e a
x         -> Res# st e a
x
{-# inline[1] (<|>) #-}

{-# RULES

"flatparse/reassoc-alt" forall l m r. (l <|> m) <|> r = l <|> (m <|> r)

#-}

instance MonadPlus (ParserT st r e) where
  mzero :: forall a. ParserT st r e a
mzero = forall (f :: * -> *) a. Alternative f => f a
Control.Applicative.empty
  {-# inline mzero #-}
  mplus :: forall a. ParserT st r e a -> ParserT st r e a -> ParserT st r e a
mplus = forall (st :: ZeroBitType) r e a.
ParserT st r e a -> ParserT st r e a -> ParserT st r e a
(<|>)
  {-# inline mplus #-}

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

-- | Primitive parser result wrapped with a state token.
--
-- You should rarely need to manipulate values of this type directly. Use the
-- provided bidirectional pattern synonyms 'OK#', 'Fail#' and 'Err#'.
type Res# (st :: ZeroBitType) e a =
  (# st, ResI# e a #)

-- | Primitive parser result.
type ResI# e a =
  (#
    (# a, Addr#, Int# #)
  | (# #)
  | (# e #)
  #)

-- | 'Res#' constructor for a successful parse.
--   Contains the return value, a pointer to the rest of the input buffer, and
--   the next 'Int' state, plus a state token.
pattern OK# :: (st :: ZeroBitType) -> a -> Addr# -> Int# -> Res# st e a
pattern $bOK# :: forall (st :: ZeroBitType) a e.
st -> a -> Addr# -> Int# -> Res# st e a
$mOK# :: forall {r} {st :: ZeroBitType} {a} {e}.
Res# st e a -> (st -> a -> Addr# -> Int# -> r) -> ((# #) -> r) -> r
OK# st a s n = (# st, (# (# a, s, n #) | | #) #)

-- | 'Res#' constructor for recoverable failure.
--   Contains only a state token.
pattern Fail# :: (st :: ZeroBitType) -> Res# st e a
pattern $bFail# :: forall (st :: ZeroBitType) e a. st -> Res# st e a
$mFail# :: forall {r} {st :: ZeroBitType} {e} {a}.
Res# st e a -> (st -> r) -> ((# #) -> r) -> r
Fail# st = (# st, (# | (# #) | #) #)

-- | 'Res#' constructor for errors which are by default non-recoverable.
--    Contains the error, plus a state token.
pattern Err# :: (st :: ZeroBitType) -> e -> Res# st e a
pattern $bErr# :: forall (st :: ZeroBitType) e a. st -> e -> Res# st e a
$mErr# :: forall {r} {st :: ZeroBitType} {e} {a}.
Res# st e a -> (st -> e -> r) -> ((# #) -> r) -> r
Err# st e = (# st, (# | | (# e #) #) #)
{-# complete OK#, Fail#, Err# #-}