{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE OverloadedLabels #-}
module Language.EO.Phi.Normalize (
normalizeObject,
normalize,
peelObject,
unpeelObject,
) where
import Control.Monad.State
import Data.Maybe (fromMaybe)
import Data.Generics.Labels ()
import GHC.Generics (Generic)
import Language.EO.Phi.Rules.Common (lookupBinding, objectBindings)
import Language.EO.Phi.Syntax (desugar, errorExpectedDesugaredObject)
import Language.EO.Phi.Syntax.Abs
data Context = Context
{ Context -> [Binding]
globalObject :: [Binding]
, Context -> [Binding]
thisObject :: [Binding]
}
deriving ((forall x. Context -> Rep Context x)
-> (forall x. Rep Context x -> Context) -> Generic Context
forall x. Rep Context x -> Context
forall x. Context -> Rep Context x
forall a.
(forall x. a -> Rep a x) -> (forall x. Rep a x -> a) -> Generic a
$cfrom :: forall x. Context -> Rep Context x
from :: forall x. Context -> Rep Context x
$cto :: forall x. Rep Context x -> Context
to :: forall x. Rep Context x -> Context
Generic)
normalize :: Program -> Program
normalize :: Program -> Program
normalize (Program [Binding]
bindings) = State Context Program -> Context -> Program
forall s a. State s a -> s -> a
evalState ([Binding] -> Program
Program ([Binding] -> Program)
-> (Object -> [Binding]) -> Object -> Program
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Object -> [Binding]
objectBindings (Object -> Program)
-> StateT Context Identity Object -> State Context Program
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Object -> StateT Context Identity Object
normalizeObject ([Binding] -> Object
Formation [Binding]
bindings)) Context
context
where
context :: Context
context =
Context
{ globalObject :: [Binding]
globalObject = [Binding]
bindings
, thisObject :: [Binding]
thisObject = [Binding]
bindings
}
normalizeObject :: Object -> State Context Object
normalizeObject :: Object -> StateT Context Identity Object
normalizeObject Object
object = do
[Binding]
this <- (Context -> [Binding]) -> StateT Context Identity [Binding]
forall s (m :: * -> *) a. MonadState s m => (s -> a) -> m a
gets Context -> [Binding]
thisObject
case Object
object of
ObjectDispatch Object
ThisObject Attribute
a -> Object -> StateT Context Identity Object
forall a. a -> StateT Context Identity a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Object -> StateT Context Identity Object)
-> Object -> StateT Context Identity Object
forall a b. (a -> b) -> a -> b
$ Object -> Maybe Object -> Object
forall a. a -> Maybe a -> a
fromMaybe Object
object (Attribute -> [Binding] -> Maybe Object
lookupBinding Attribute
a [Binding]
this)
Object
_ -> Object -> StateT Context Identity Object
forall a. a -> StateT Context Identity a
forall (f :: * -> *) a. Applicative f => a -> f a
pure Object
object
peelObject :: Object -> PeeledObject
peelObject :: Object -> PeeledObject
peelObject = \case
Formation [Binding]
bindings -> ObjectHead -> [ObjectAction] -> PeeledObject
PeeledObject ([Binding] -> ObjectHead
HeadFormation [Binding]
bindings) []
Application Object
object [Binding]
bindings -> Object -> PeeledObject
peelObject Object
object PeeledObject -> ObjectAction -> PeeledObject
`followedBy` [Binding] -> ObjectAction
ActionApplication [Binding]
bindings
ObjectDispatch Object
object Attribute
attr -> Object -> PeeledObject
peelObject Object
object PeeledObject -> ObjectAction -> PeeledObject
`followedBy` Attribute -> ObjectAction
ActionDispatch Attribute
attr
Object
GlobalObject -> ObjectHead -> [ObjectAction] -> PeeledObject
PeeledObject ObjectHead
HeadGlobal []
obj :: Object
obj@Object
GlobalObjectPhiOrg -> Object -> PeeledObject
peelObject (Object -> Object
desugar Object
obj)
Object
ThisObject -> ObjectHead -> [ObjectAction] -> PeeledObject
PeeledObject ObjectHead
HeadThis []
Object
Termination -> ObjectHead -> [ObjectAction] -> PeeledObject
PeeledObject ObjectHead
HeadTermination []
MetaObject ObjectMetaId
_ -> ObjectHead -> [ObjectAction] -> PeeledObject
PeeledObject ObjectHead
HeadTermination []
MetaTailContext{} -> [Char] -> PeeledObject
forall a. HasCallStack => [Char] -> a
error [Char]
"impossible"
MetaFunction MetaFunctionName
_ Object
_ -> [Char] -> PeeledObject
forall a. HasCallStack => [Char] -> a
error [Char]
"To be honest, I'm not sure what should be here"
MetaSubstThis{} -> [Char] -> PeeledObject
forall a. HasCallStack => [Char] -> a
error [Char]
"impossible"
MetaContextualize{} -> [Char] -> PeeledObject
forall a. HasCallStack => [Char] -> a
error [Char]
"impossible"
obj :: Object
obj@ConstString{} -> Object -> PeeledObject
peelObject (Object -> Object
desugar Object
obj)
obj :: Object
obj@ConstStringRaw{} -> Object -> PeeledObject
forall a. Object -> a
errorExpectedDesugaredObject Object
obj
obj :: Object
obj@ConstInt{} -> Object -> PeeledObject
peelObject (Object -> Object
desugar Object
obj)
obj :: Object
obj@ConstIntRaw{} -> Object -> PeeledObject
forall a. Object -> a
errorExpectedDesugaredObject Object
obj
obj :: Object
obj@ConstFloat{} -> Object -> PeeledObject
peelObject (Object -> Object
desugar Object
obj)
obj :: Object
obj@ConstFloatRaw{} -> Object -> PeeledObject
forall a. Object -> a
errorExpectedDesugaredObject Object
obj
where
followedBy :: PeeledObject -> ObjectAction -> PeeledObject
followedBy (PeeledObject ObjectHead
object [ObjectAction]
actions) ObjectAction
action = ObjectHead -> [ObjectAction] -> PeeledObject
PeeledObject ObjectHead
object ([ObjectAction]
actions [ObjectAction] -> [ObjectAction] -> [ObjectAction]
forall a. [a] -> [a] -> [a]
++ [ObjectAction
action])
unpeelObject :: PeeledObject -> Object
unpeelObject :: PeeledObject -> Object
unpeelObject (PeeledObject ObjectHead
head_ [ObjectAction]
actions) =
case ObjectHead
head_ of
HeadFormation [Binding]
bindings -> Object -> [ObjectAction] -> Object
go ([Binding] -> Object
Formation [Binding]
bindings) [ObjectAction]
actions
ObjectHead
HeadGlobal ->
case [ObjectAction]
actions of
ActionApplication{} : [ObjectAction]
_ -> [Char] -> Object
forall a. HasCallStack => [Char] -> a
error [Char]
"impossible: application for a global object!"
[ObjectAction]
_ -> Object -> [ObjectAction] -> Object
go Object
GlobalObject [ObjectAction]
actions
ObjectHead
HeadThis ->
case [ObjectAction]
actions of
ActionApplication{} : [ObjectAction]
_ -> [Char] -> Object
forall a. HasCallStack => [Char] -> a
error [Char]
"impossible: application for a global object!"
[ObjectAction]
_ -> Object -> [ObjectAction] -> Object
go Object
ThisObject [ObjectAction]
actions
ObjectHead
HeadTermination -> Object -> [ObjectAction] -> Object
go Object
Termination [ObjectAction]
actions
where
go :: Object -> [ObjectAction] -> Object
go = (Object -> ObjectAction -> Object)
-> Object -> [ObjectAction] -> Object
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl Object -> ObjectAction -> Object
applyAction
applyAction :: Object -> ObjectAction -> Object
applyAction Object
object = \case
ActionDispatch Attribute
attr -> Object -> Attribute -> Object
ObjectDispatch Object
object Attribute
attr
ActionApplication [Binding]
bindings -> Object -> [Binding] -> Object
Application Object
object [Binding]
bindings