module FreeC.Backend.Coq.Converter.TypeDecl
(
convertTypeComponent
, sortTypeSynDecls
, fromNonRecursive
, isTypeSynDecl
, convertTypeSynDecl
, convertDataDecls
, convertDataDecl
) where
import Control.Monad
( mapAndUnzipM )
import Control.Monad.Extra
( concatMapM )
import Data.List
( partition )
import Data.List.Extra
( concatUnzip )
import qualified Data.List.NonEmpty as NonEmpty
import Data.Maybe
( catMaybes, fromJust )
import qualified Data.Set as Set
import qualified FreeC.Backend.Coq.Base as Coq.Base
import FreeC.Backend.Coq.Converter.Arg
import FreeC.Backend.Coq.Converter.Free
import FreeC.Backend.Coq.Converter.Type
import FreeC.Backend.Coq.Converter.TypeDecl.InductionScheme
import FreeC.Backend.Coq.Converter.TypeDecl.TypeclassInstances
import qualified FreeC.Backend.Coq.Syntax as Coq
import FreeC.Environment
import FreeC.Environment.Fresh
( freshArgPrefix, freshCoqIdent )
import FreeC.Environment.Renamer
( renameAndDefineTypeVar )
import FreeC.IR.DependencyGraph
import qualified FreeC.IR.Syntax as IR
import FreeC.IR.TypeSynExpansion
import FreeC.Monad.Converter
import FreeC.Monad.Reporter
import FreeC.Pretty
convertTypeComponent :: DependencyComponent IR.TypeDecl
-> Converter ([Coq.Sentence], [Coq.Sentence])
convertTypeComponent (NonRecursive decl)
| isTypeSynDecl decl = flip (,) [] <$> convertTypeSynDecl decl
| otherwise = convertDataDecls [decl]
convertTypeComponent (Recursive decls) = do
let (typeSynDecls, dataDecls) = partition isTypeSynDecl decls
typeSynDeclQNames = Set.fromList
(map IR.typeDeclQName typeSynDecls)
sortedTypeSynDecls <- sortTypeSynDecls typeSynDecls
expandedDataDecls <- mapM
(expandAllTypeSynonymsInDeclWhere (`Set.member` typeSynDeclQNames))
dataDecls
(dataDecls', qualSmartConDecls) <- convertDataDecls expandedDataDecls
typeSynDecls' <- concatMapM convertTypeSynDecl sortedTypeSynDecls
return (dataDecls' ++ typeSynDecls', qualSmartConDecls)
sortTypeSynDecls :: [IR.TypeDecl] -> Converter [IR.TypeDecl]
sortTypeSynDecls = mapM fromNonRecursive . typeDependencyComponents
fromNonRecursive :: DependencyComponent IR.TypeDecl -> Converter IR.TypeDecl
fromNonRecursive (NonRecursive decl) = return decl
fromNonRecursive (Recursive decls) = reportFatal
$ Message (IR.typeDeclSrcSpan (head decls)) Error
$ "Type synonym declarations form a cycle: "
++ showPretty (map IR.typeDeclIdent decls)
isTypeSynDecl :: IR.TypeDecl -> Bool
isTypeSynDecl (IR.TypeSynDecl _ _ _ _) = True
isTypeSynDecl (IR.DataDecl _ _ _ _) = False
convertTypeSynDecl :: IR.TypeDecl -> Converter [Coq.Sentence]
convertTypeSynDecl decl@(IR.TypeSynDecl _ _ typeVarDecls typeExpr)
= localEnv $ do
let name = IR.typeDeclQName decl
Just qualid <- inEnv $ lookupIdent IR.TypeScope name
typeVarDecls' <- convertTypeVarDecls Coq.Explicit typeVarDecls
typeExpr' <- convertType' typeExpr
return [ Coq.definitionSentence qualid
(genericArgDecls Coq.Explicit ++ typeVarDecls')
(Just Coq.sortType) typeExpr'
]
convertTypeSynDecl (IR.DataDecl _ _ _ _)
= error "convertTypeSynDecl: Data type declaration not allowed."
convertDataDecls :: [IR.TypeDecl] -> Converter ([Coq.Sentence], [Coq.Sentence])
convertDataDecls dataDecls = do
(indBodies, extraSentences') <- mapAndUnzipM convertDataDecl dataDecls
let (extraSentences, qualSmartConDecls) = concatUnzip extraSentences'
instances <- generateTypeclassInstances dataDecls
return
( Coq.comment ("Data type declarations for "
++ showPretty (map IR.typeDeclName dataDecls))
: Coq.unsetOption (Just Coq.Local) "Elimination Schemes"
: Coq.InductiveSentence (Coq.Inductive (NonEmpty.fromList indBodies) [])
: Coq.setOption (Just Coq.Local) "Elimination Schemes" Nothing
: extraSentences ++ instances
, qualSmartConDecls
)
convertDataDecl
:: IR.TypeDecl -> Converter (Coq.IndBody, ([Coq.Sentence], [Coq.Sentence]))
convertDataDecl
dataDecl@(IR.DataDecl _ (IR.DeclIdent _ name) typeVarDecls conDecls) = do
(body, argumentsSentences) <- generateBodyAndArguments
(smartConDecls, qualSmartConDecls)
<- concatUnzip <$> mapM generateSmartConDecl conDecls
inductionScheme <- generateInductionScheme dataDecl
return ( body
, ( Coq.commentedSentences
("Arguments sentences for " ++ showPretty (IR.toUnQual name))
argumentsSentences
++ Coq.commentedSentences
("Induction scheme for " ++ showPretty (IR.toUnQual name))
inductionScheme
++ Coq.commentedSentences
("Smart constructors for " ++ showPretty (IR.toUnQual name))
smartConDecls
, Coq.commentedSentences ("Qualified smart constructors for "
++ showPretty (IR.toUnQual name))
qualSmartConDecls
)
)
where
generateBodyAndArguments :: Converter (Coq.IndBody, [Coq.Sentence])
generateBodyAndArguments = localEnv $ do
Just qualid <- inEnv $ lookupIdent IR.TypeScope name
typeVarDecls' <- convertTypeVarDecls Coq.Explicit typeVarDecls
conDecls' <- mapM convertConDecl conDecls
argumentsSentences <- mapM generateArgumentsSentence conDecls
return ( Coq.IndBody qualid (genericArgDecls Coq.Explicit ++ typeVarDecls')
Coq.sortType conDecls'
, argumentsSentences
)
convertConDecl
:: IR.ConDecl -> Converter (Coq.Qualid, [Coq.Binder], Maybe Coq.Term)
convertConDecl (IR.ConDecl _ (IR.DeclIdent _ conName) args) = do
Just conQualid <- inEnv $ lookupIdent IR.ValueScope conName
Just returnType <- inEnv $ lookupReturnType IR.ValueScope conName
args' <- mapM convertType args
returnType' <- convertType' returnType
return (conQualid, [], Just (args' `Coq.arrows` returnType'))
generateArgumentsSentence :: IR.ConDecl -> Converter Coq.Sentence
generateArgumentsSentence (IR.ConDecl _ (IR.DeclIdent _ conName) _) = do
Just qualid <- inEnv $ lookupIdent IR.ValueScope conName
let typeVarNames = map IR.typeVarDeclQName typeVarDecls
typeVarQualids <- mapM (inEnv . lookupIdent IR.TypeScope) typeVarNames
return (Coq.ArgumentsSentence
(Coq.Arguments Nothing qualid
[Coq.ArgumentSpec Coq.ArgMaximal (Coq.Ident typeVarQualid) Nothing
| typeVarQualid
<- map fst Coq.Base.freeArgs ++ catMaybes typeVarQualids
]))
generateSmartConDecl
:: IR.ConDecl -> Converter ([Coq.Sentence], [Coq.Sentence])
generateSmartConDecl (IR.ConDecl _ declIdent argTypes) = localEnv $ do
let conName = IR.declIdentName declIdent
mbModName <- inEnv $ lookupModName IR.ValueScope conName
Just qualid <- inEnv $ lookupIdent IR.ValueScope conName
Just smartQualid <- inEnv $ lookupSmartIdent conName
Just returnType <- inEnv $ lookupReturnType IR.ValueScope conName
constrArgIdents <- mapM (const $ freshCoqIdent freshArgPrefix) argTypes
let Just unqualIdent = Coq.unpackQualid smartQualid
typeVarQualids <- mapM
(\(IR.TypeVarDecl srcSpan ident) -> renameAndDefineTypeVar srcSpan ident)
typeVarDecls
let typeVarIdents = map (fromJust . Coq.unpackQualid) typeVarQualids
returnType' <- convertType' returnType
return
( generateSmartConDecl' unqualIdent qualid typeVarIdents constrArgIdents
returnType'
, case mbModName of
Just modName -> generateSmartConDecl' (modName ++ '.' : unqualIdent)
qualid typeVarIdents constrArgIdents returnType'
Nothing -> []
)
generateSmartConDecl'
:: String
-> Coq.Qualid
-> [String]
-> [String]
-> Coq.Term
-> [Coq.Sentence]
generateSmartConDecl' smartIdent constr typeVarIdents constrArgIdents
expReturnType = [ Coq.notationSentence lhs rhs mods
, Coq.notationSentence expLhs expRhs expMods
]
where
freeArgIdents = map (fromJust . Coq.unpackQualid . fst) Coq.Base.freeArgs
freeArgs = map (Coq.Qualid . fst) Coq.Base.freeArgs
returnType = case expReturnType of
(Coq.App tcon (shape NonEmpty.:| pos : tvars))
| length tvars == length typeVarIdents -> Coq.App tcon
(shape NonEmpty.:| pos
: map (Coq.PosArg . const Coq.Underscore) tvars)
_ -> Coq.Underscore
argIdents = freeArgIdents ++ constrArgIdents
args = freeArgs
++ map (const Coq.Underscore) typeVarIdents
++ map (Coq.Qualid . Coq.bare) constrArgIdents
lhs = Coq.nSymbol smartIdent NonEmpty.:| map Coq.nIdent argIdents
rhs = Coq.explicitApp Coq.Base.freePureCon
$ freeArgs ++ [returnType, Coq.explicitApp constr args]
mods = [ Coq.sModLevel 10
, Coq.sModIdentLevel (NonEmpty.fromList argIdents) (Just 9)
]
expArgIdents = freeArgIdents ++ typeVarIdents ++ constrArgIdents
expArgs = map (Coq.Qualid . Coq.bare) expArgIdents
expLhs = Coq.nSymbol ('@' : smartIdent)
NonEmpty.:| map Coq.nIdent expArgIdents
expRhs = Coq.explicitApp Coq.Base.freePureCon
$ freeArgs ++ [expReturnType, Coq.explicitApp constr expArgs]
expMods
= [ Coq.SModOnlyParsing
, Coq.sModLevel 10
, Coq.sModIdentLevel (NonEmpty.fromList expArgIdents) (Just 9)
]
convertDataDecl (IR.TypeSynDecl _ _ _ _)
= error "convertDataDecl: Type synonym not allowed."