module FreeC.Backend.Coq.Syntax
( module Language.Coq.Gallina
, comment
, commentedSentences
, unpackComment
, blankProof
, ident
, bare
, qualified
, unpackQualid
, access
, app
, explicitApp
, arrows
, fun
, inferredFun
, typedBinder
, typedBinder'
, variable
, context
, definitionSentence
, notationSentence
, nSymbol
, nIdent
, sModLevel
, sModIdentLevel
, sortType
, string
, equation
, match
, equals
, notEquals
, conj
, disj
, setOption
, unsetOption
, requireImportFrom
, requireExportFrom
, requireFrom
, moduleImport
, moduleExport
) where
import Prelude hiding ( Num )
import Data.Composition ( (.:) )
import qualified Data.List.NonEmpty as NonEmpty
import qualified Data.Text as Text
import Language.Coq.Gallina
comment :: String -> Sentence
comment = CommentSentence . Comment . Text.pack
commentedSentences :: String -> [Sentence] -> [Sentence]
commentedSentences _ [] = []
commentedSentences str sentences = comment str : sentences
unpackComment :: Comment -> String
unpackComment (Comment c) = Text.unpack c
blankProof :: Proof
blankProof = ProofAdmitted (Text.pack " (* FILL IN HERE *)")
ident :: String -> Ident
ident = Text.pack
bare :: String -> Qualid
bare = Bare . ident
qualified :: String -> String -> Qualid
qualified modName name = Qualified (ident modName) (ident name)
unpackQualid :: Qualid -> Maybe String
unpackQualid (Bare text) = Just (Text.unpack text)
unpackQualid (Qualified _ _) = Nothing
access :: ModuleIdent -> Ident -> Ident
access modName name = Text.append modName (Text.cons '.' name)
app :: Term -> [Term] -> Term
app func [] = func
app (App func args) args' = App func
(args <> NonEmpty.fromList (map PosArg args'))
app func args = App func (NonEmpty.fromList (map PosArg args))
explicitApp :: Qualid -> [Term] -> Term
explicitApp qualid [] = Qualid qualid
explicitApp qualid typeArgs = ExplicitApp qualid typeArgs
arrows :: [Term]
-> Term
-> Term
arrows args returnType = foldr Arrow returnType args
fun :: [Qualid] -> [Maybe Term] -> Term -> Term
fun args argTypes = Fun (NonEmpty.fromList binders)
where
argNames :: [Name]
argNames = map Ident args
binders :: [Binder]
binders = zipWith makeBinder argTypes argNames
makeBinder :: Maybe Term -> Name -> Binder
makeBinder Nothing = Inferred Explicit
makeBinder (Just t) = flip (Typed Ungeneralizable Explicit) t . return
inferredFun :: [Qualid] -> Term -> Term
inferredFun = flip fun (repeat Nothing)
typedBinder :: Generalizability -> Explicitness -> [Qualid] -> Term -> Binder
typedBinder generalizability explicitness
= Typed generalizability explicitness . NonEmpty.fromList . map Ident
typedBinder' :: Generalizability -> Explicitness -> Qualid -> Term -> Binder
typedBinder' generalizability explicitness term
= typedBinder generalizability explicitness [term]
variable :: [Qualid] -> Term -> Sentence
variable
= AssumptionSentence . Assumption Variable .: Assums . NonEmpty.fromList
context :: Binder -> Sentence
context = ContextSentence . NonEmpty.fromList . (: [])
definitionSentence
:: Qualid
-> [Binder]
-> Maybe Term
-> Term
-> Sentence
definitionSentence qualid binders returnType term = DefinitionSentence
(DefinitionDef Global qualid binders returnType term)
notationSentence
:: NonEmpty.NonEmpty NotationToken
-> Term
-> [SyntaxModifier]
-> Sentence
notationSentence tokens rhs smods = NotationSentence
(NotationDefinition tokens rhs smods)
nSymbol :: String -> NotationToken
nSymbol = NSymbol . Text.pack
nIdent :: String -> NotationToken
nIdent = NIdent . ident
sModLevel :: Num -> SyntaxModifier
sModLevel = SModLevel . Level
sModIdentLevel :: NonEmpty.NonEmpty String -> Maybe Num -> SyntaxModifier
sModIdentLevel idents (Just lvl) = SModIdentLevel (NonEmpty.map ident idents)
(ExplicitLevel $ Level lvl)
sModIdentLevel idents Nothing = SModIdentLevel (NonEmpty.map ident idents)
NextLevel
sortType :: Term
sortType = Sort Type
string :: String -> Term
string = String . Text.pack
equation :: Pattern -> Term -> Equation
equation = Equation . return . MultPattern . return
match :: Term -> [Equation] -> Term
match value = Match (return item) Nothing
where
item :: MatchItem
item = MatchItem value Nothing Nothing
equals :: Term -> Term -> Term
equals t1 t2 = app (Qualid (bare "op_=__")) [t1, t2]
notEquals :: Term -> Term -> Term
notEquals t1 t2 = app (Qualid (bare "op_<>__")) [t1, t2]
conj :: Term -> Term -> Term
conj t1 t2 = app (Qualid (bare "op_/\\__")) [t1, t2]
disj :: Term -> Term -> Term
disj t1 t2 = app (Qualid (bare "op_\\/__")) [t1, t2]
setOption :: Maybe Locality -> String -> Maybe (Either Num String) -> Sentence
setOption mbLoc opt mbArg = OptionSentence
$ SetOption mbLoc (Text.pack opt) mbArg'
where
mbArg' = case mbArg of
Nothing -> Nothing
(Just (Left num)) -> Just (OVNum num)
(Just (Right str)) -> Just (OVText (Text.pack str))
unsetOption :: Maybe Locality -> String -> Sentence
unsetOption mbLoc opt = OptionSentence $ UnsetOption mbLoc (Text.pack opt)
requireImportFrom :: ModuleIdent -> [ModuleIdent] -> Sentence
requireImportFrom library modules = ModuleSentence
(Require (Just library) (Just Import) (NonEmpty.fromList modules))
requireExportFrom :: ModuleIdent -> [ModuleIdent] -> Sentence
requireExportFrom library modules = ModuleSentence
(Require (Just library) (Just Export) (NonEmpty.fromList modules))
requireFrom :: ModuleIdent -> [ModuleIdent] -> Sentence
requireFrom library modules = ModuleSentence
(Require (Just library) Nothing (NonEmpty.fromList modules))
moduleImport :: [ModuleIdent] -> Sentence
moduleImport modules = ModuleSentence
(ModuleImport Import (NonEmpty.fromList modules))
moduleExport :: [ModuleIdent] -> Sentence
moduleExport modules = ModuleSentence
(ModuleImport Export (NonEmpty.fromList modules))