205 lines
4.8 KiB
Haskell
Raw Normal View History

2020-06-04 13:48:04 +04:00
{- | /The/ scope resolution system.
-}
2020-06-04 13:48:04 +04:00
module AST.Scope
( -- * Monad
ScopeM
, evalScopeM
2020-06-09 15:56:11 +04:00
, pinEnv
2020-06-04 13:48:04 +04:00
-- * Scope
, Env(..)
, ScopedDecl(..)
, Kind(..)
2020-06-09 15:56:11 +04:00
, HasEnv(..)
, lookupEnv
2020-06-04 13:48:04 +04:00
-- * Methods
, enter
, leave
, define
, defType
, def
)
where
import Control.Monad.State
import Data.Text (Text)
import Range
import AST.Types
2020-06-01 18:17:33 +04:00
import Tree
import Comment
2020-06-01 18:17:33 +04:00
import Pretty
2020-06-09 15:56:11 +04:00
import Product
2020-06-04 17:40:38 +04:00
-- | Scope-holding monad.
type ScopeM = State [Env]
2020-06-04 17:40:38 +04:00
-- | Run the computation with scope starting from empty scope.
evalScopeM :: ScopeM a -> a
evalScopeM action = evalState action [Env []]
-- | The environment.
newtype Env = Env
{ _eDecls :: [ScopedDecl]
}
deriving newtype (Semigroup, Monoid)
2020-06-09 15:56:11 +04:00
deriving Show via PP Env
instance Pretty Env where
pp = vcat . map pp . _eDecls
2020-06-04 17:40:38 +04:00
-- | The type/value declaration.
data ScopedDecl = ScopedDecl
2020-06-04 16:34:58 +04:00
{ _sdName :: (Pascal ())
, _sdOrigin :: Range
, _sdBody :: Maybe Range
2020-06-01 18:17:33 +04:00
, _sdType :: Maybe (Either (Pascal ()) Kind)
}
2020-06-09 15:56:11 +04:00
instance Pretty ScopedDecl where
pp (ScopedDecl n o b t) = pp o <+> "-" <+> (pp n <> ":") <+> maybe "?" (either pp pp) t <+> "=" <+> pp o
2020-06-04 17:40:38 +04:00
-- | The kind.
data Kind = Star
2020-06-09 15:56:11 +04:00
deriving Show via PP Kind
instance Pretty Kind where
pp _ = "*"
lookupEnv :: Pascal () -> Env -> Maybe ScopedDecl
lookupEnv name = go . _eDecls
where
go (sd@(ScopedDecl {_sdName}) : rest)
| ppToText _sdName == ppToText name = Just sd
| otherwise = go rest
go _ = Nothing
2020-06-04 17:40:38 +04:00
-- | Make a new scope out of enclosing parent one.
enter :: ScopeM ()
enter = modify \(a : b) -> a : a : b
-- | Leave current scope, return to parent one.
leave :: ScopeM ()
leave = modify tail
-- | Add a declaration to the current scope.
2020-06-04 17:16:04 +04:00
define :: ScopedDecl -> ScopeM ()
define d = modify \(Env a : b) -> Env (d : a) : b
2020-06-01 18:17:33 +04:00
2020-06-04 17:40:38 +04:00
-- | Add a type declaration to the current scope.
2020-06-01 18:17:33 +04:00
defType :: HasRange a => Pascal a -> Kind -> Pascal a -> ScopeM ()
defType name kind body = do
define $ ScopedDecl
2020-06-04 16:34:58 +04:00
(void name)
(getRange $ infoOf name)
(Just $ getRange $ infoOf body)
2020-06-01 18:17:33 +04:00
(Just (Right kind))
2020-06-04 17:40:38 +04:00
-- | Add a value declaration to the current scope.
def
2020-06-01 18:17:33 +04:00
:: HasRange a
=> Pascal a
-> Maybe (Pascal a)
-> Maybe (Pascal a)
-> ScopeM ()
2020-06-01 18:17:33 +04:00
def name ty body = do
define $ ScopedDecl
2020-06-04 16:34:58 +04:00
(void name)
(getRange $ infoOf name)
((getRange . infoOf) <$> body)
2020-06-01 18:17:33 +04:00
((Left . void) <$> ty)
instance UpdateOver ScopeM Contract (Pascal a)
instance HasRange a => UpdateOver ScopeM Declaration (Pascal a) where
before = \case
TypeDecl ty body -> defType ty Star body
_ -> skip
instance HasRange a => UpdateOver ScopeM Binding (Pascal a) where
before = \case
2020-06-04 17:16:04 +04:00
Function recur name _args ty body -> do
when recur do
2020-06-01 18:17:33 +04:00
def name (Just ty) (Just body)
enter
2020-06-01 18:17:33 +04:00
_ -> enter
2020-06-01 18:17:33 +04:00
after = \case
Irrefutable name body -> do leave; def name Nothing (Just body)
Var name ty body -> do leave; def name (Just ty) (Just body)
Const name ty body -> do leave; def name (Just ty) (Just body)
2020-06-04 17:16:04 +04:00
Function recur name _args ty body -> do
2020-06-01 18:17:33 +04:00
leave
unless recur do
2020-06-01 18:17:33 +04:00
def name (Just ty) (Just body)
instance HasRange a => UpdateOver ScopeM VarDecl (Pascal a) where
after (Decl _ name ty) = def name (Just ty) Nothing
instance UpdateOver ScopeM Mutable (Pascal a)
2020-06-09 15:56:11 +04:00
instance UpdateOver ScopeM Type (Pascal a)
2020-06-01 18:17:33 +04:00
instance UpdateOver ScopeM Variant (Pascal a)
2020-06-09 15:56:11 +04:00
instance UpdateOver ScopeM TField (Pascal a)
2020-06-01 18:17:33 +04:00
instance HasRange a => UpdateOver ScopeM Expr (Pascal a) where
before = \case
Let {} -> enter
Lambda {} -> enter
ForLoop k _ _ _ -> do
enter
def k Nothing Nothing
ForBox k mv _ _ _ -> do
enter
def k Nothing Nothing
maybe skip (\v -> def v Nothing Nothing) mv
_ -> skip
after = \case
Let {} -> leave
Lambda {} -> leave
ForLoop {} -> leave
ForBox {} -> leave
_ -> skip
instance HasRange a => UpdateOver ScopeM Alt (Pascal a) where
before _ = enter
after _ = leave
2020-06-09 15:56:11 +04:00
instance UpdateOver ScopeM LHS (Pascal a)
instance UpdateOver ScopeM MapBinding (Pascal a)
instance UpdateOver ScopeM Assignment (Pascal a)
2020-06-01 18:17:33 +04:00
instance UpdateOver ScopeM FieldAssignment (Pascal a)
2020-06-09 15:56:11 +04:00
instance UpdateOver ScopeM Constant (Pascal a)
2020-06-01 18:17:33 +04:00
instance HasRange a => UpdateOver ScopeM Pattern (Pascal a) where
before = \case
IsVar n -> def n Nothing Nothing
_ -> skip
instance UpdateOver ScopeM QualifiedName (Pascal a)
2020-06-09 15:56:11 +04:00
instance UpdateOver ScopeM Path (Pascal a)
instance UpdateOver ScopeM Name (Pascal a)
2020-06-01 18:17:33 +04:00
2020-06-09 15:56:11 +04:00
class HasEnv a where
getEnv :: a -> Env
2020-06-01 18:17:33 +04:00
2020-06-09 15:56:11 +04:00
instance HasEnv Env where
getEnv = id
2020-06-01 18:17:33 +04:00
2020-06-09 15:56:11 +04:00
instance Contains Env xs => HasEnv (Product xs) where
getEnv = getElem
2020-06-01 18:17:33 +04:00
data Scope = Scope { unScope :: [Text] }
2020-06-01 18:17:33 +04:00
instance HasComments Scope where
getComments = unScope
2020-06-01 18:17:33 +04:00
2020-06-09 15:56:11 +04:00
pinEnv :: Product xs -> ScopeM (Product (Env : xs))
pinEnv xs = (`Cons` xs) <$> gets head