2020-05-21 23:28:26 +04:00
|
|
|
|
2020-06-04 13:48:04 +04:00
|
|
|
{- | /The/ scope resolution system.
|
2020-05-21 23:28:26 +04:00
|
|
|
-}
|
|
|
|
|
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
|
2020-05-21 23:28:26 +04:00
|
|
|
|
|
|
|
import Control.Monad.State
|
|
|
|
|
|
|
|
import Data.Text (Text)
|
|
|
|
|
|
|
|
import Range
|
|
|
|
import AST.Types
|
2020-06-01 18:17:33 +04:00
|
|
|
import Tree
|
2020-06-04 19:15:14 +04:00
|
|
|
import Comment
|
2020-06-01 18:17:33 +04:00
|
|
|
import Pretty
|
2020-06-09 15:56:11 +04:00
|
|
|
import Product
|
2020-05-21 23:28:26 +04:00
|
|
|
|
2020-06-04 17:40:38 +04:00
|
|
|
-- | Scope-holding monad.
|
2020-05-21 23:28:26 +04:00
|
|
|
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.
|
2020-05-21 23:28:26 +04:00
|
|
|
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-05-21 23:28:26 +04:00
|
|
|
|
2020-06-04 17:40:38 +04:00
|
|
|
-- | The type/value declaration.
|
2020-05-21 23:28:26 +04:00
|
|
|
data ScopedDecl = ScopedDecl
|
2020-06-04 16:34:58 +04:00
|
|
|
{ _sdName :: (Pascal ())
|
|
|
|
, _sdOrigin :: Range
|
2020-05-21 23:28:26 +04:00
|
|
|
, _sdBody :: Maybe Range
|
2020-06-01 18:17:33 +04:00
|
|
|
, _sdType :: Maybe (Either (Pascal ()) Kind)
|
2020-05-21 23:28:26 +04:00
|
|
|
}
|
|
|
|
|
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.
|
2020-05-21 23:28:26 +04:00
|
|
|
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-05-21 23:28:26 +04:00
|
|
|
|
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-05-21 23:28:26 +04:00
|
|
|
|
2020-06-04 17:40:38 +04:00
|
|
|
-- | Add a value declaration to the current scope.
|
2020-05-21 23:28:26 +04:00
|
|
|
def
|
2020-06-01 18:17:33 +04:00
|
|
|
:: HasRange a
|
|
|
|
=> Pascal a
|
|
|
|
-> Maybe (Pascal a)
|
|
|
|
-> Maybe (Pascal a)
|
2020-05-21 23:28:26 +04:00
|
|
|
-> ScopeM ()
|
2020-06-01 18:17:33 +04:00
|
|
|
def name ty body = do
|
2020-05-21 23:28:26 +04:00
|
|
|
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
|
2020-05-21 23:28:26 +04:00
|
|
|
when recur do
|
2020-06-01 18:17:33 +04:00
|
|
|
def name (Just ty) (Just body)
|
|
|
|
enter
|
2020-05-21 23:28:26 +04:00
|
|
|
|
2020-06-01 18:17:33 +04:00
|
|
|
_ -> enter
|
2020-05-21 23:28:26 +04:00
|
|
|
|
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
|
2020-05-21 23:28:26 +04:00
|
|
|
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
|
|
|
|
2020-06-07 23:51:44 +04:00
|
|
|
data Scope = Scope { unScope :: [Text] }
|
2020-06-01 18:17:33 +04:00
|
|
|
|
|
|
|
instance HasComments Scope where
|
2020-06-07 23:51:44 +04:00
|
|
|
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
|