{-# LANGUAGE FlexibleInstances #-}
module FreeC.IR.Syntax.Module where
import FreeC.IR.SrcSpan
import FreeC.IR.Syntax.FuncDecl
import FreeC.IR.Syntax.Name
import FreeC.IR.Syntax.Pragma
import FreeC.IR.Syntax.TypeDecl
import FreeC.Pretty
data ModuleOf contents = ModuleOf
{ modSrcSpan :: SrcSpan
, modName :: ModName
, modImports :: [ImportDecl]
, modPragmas :: [Pragma]
, modContents :: contents
}
deriving ( Eq, Show )
modWithContents
:: (contents -> contents') -> ModuleOf contents -> ModuleOf contents'
modWithContents f ast = ast { modContents = f (modContents ast) }
type Module = ModuleOf [TopLevelDecl]
instance Pretty contents => Pretty (ModuleOf [contents]) where
pretty ast = prettyString "module"
<+> prettyString (modName ast)
<+> prettyString "where"
<$$> prettySeparated (semi <> line) (map pretty (modContents ast))
data ImportDecl
= ImportDecl { importSrcSpan :: SrcSpan, importName :: ModName }
deriving ( Eq, Show )
instance Pretty ImportDecl where
pretty decl = prettyString "import" <+> prettyString (importName decl)
data TopLevelDecl
= TopLevelTypeDecl { topLevelTypeDecl :: TypeDecl }
| TopLevelTypeSig { topLevelTypeSig :: TypeSig }
| TopLevelFuncDecl { topLevelFuncDecl :: FuncDecl }
deriving ( Eq, Show )
isTopLevelTypeDecl :: TopLevelDecl -> Bool
isTopLevelTypeDecl TopLevelTypeDecl {} = True
isTopLevelTypeDecl _ = False
isTopLevelTypeSig :: TopLevelDecl -> Bool
isTopLevelTypeSig TopLevelTypeSig {} = True
isTopLevelTypeSig _ = False
isTopLevelFuncDecl :: TopLevelDecl -> Bool
isTopLevelFuncDecl TopLevelFuncDecl {} = True
isTopLevelFuncDecl _ = False
instance Pretty TopLevelDecl where
pretty (TopLevelTypeDecl typeDecl) = pretty typeDecl
pretty (TopLevelTypeSig typeSig) = pretty typeSig
pretty (TopLevelFuncDecl funcDecl) = pretty funcDecl
modTypeDecls :: Module -> [TypeDecl]
modTypeDecls = map topLevelTypeDecl . filter isTopLevelTypeDecl . modContents
modWithTypeDecls :: [TypeDecl] -> Module -> Module
modWithTypeDecls decls = modWithContents $ \contents ->
map TopLevelTypeDecl decls ++ filter (not . isTopLevelTypeDecl) contents
modTypeSigs :: Module -> [TypeSig]
modTypeSigs = map topLevelTypeSig . filter isTopLevelTypeSig . modContents
modWithTypeSigs :: [TypeSig] -> Module -> Module
modWithTypeSigs decls = modWithContents $ \contents -> map TopLevelTypeSig decls
++ filter (not . isTopLevelTypeSig) contents
modFuncDecls :: Module -> [FuncDecl]
modFuncDecls = map topLevelFuncDecl . filter isTopLevelFuncDecl . modContents
modWithFuncDecls :: [FuncDecl] -> Module -> Module
modWithFuncDecls decls = modWithContents $ \contents ->
map TopLevelFuncDecl decls ++ filter (not . isTopLevelFuncDecl) contents