module FreeC.Backend.Coq.Converter.FuncDecl.Rec.WithHelpers
( convertRecFuncDeclsWithHelpers
, convertRecFuncDeclsWithHelpers'
) where
import Control.Monad
( forM, join, mapAndUnzipM )
import Data.List
( delete, elemIndex, partition )
import qualified Data.List.NonEmpty as NonEmpty
import qualified Data.Map.Strict as Map
import Data.Maybe
( fromJust, isJust )
import Data.Set ( Set )
import qualified Data.Set as Set
import FreeC.Backend.Coq.Analysis.DecreasingArguments
import FreeC.Backend.Coq.Converter.Expr
import FreeC.Backend.Coq.Converter.FuncDecl.Common
import FreeC.Backend.Coq.Converter.FuncDecl.NonRec
import qualified FreeC.Backend.Coq.Syntax as Coq
import FreeC.Environment
import FreeC.Environment.Entry
import FreeC.Environment.Fresh
import FreeC.Environment.Renamer
import FreeC.IR.Inlining
import FreeC.IR.Reference ( freeVarSet )
import FreeC.IR.SrcSpan
import FreeC.IR.Subst
import FreeC.IR.Subterm
import qualified FreeC.IR.Syntax as IR
import FreeC.Monad.Converter
import FreeC.Pretty
import FreeC.Util.Predicate
convertRecFuncDeclsWithHelpers :: [IR.FuncDecl] -> Converter [Coq.Sentence]
convertRecFuncDeclsWithHelpers decls = do
(helperDecls', mainDecls') <- convertRecFuncDeclsWithHelpers' decls
return (Coq.comment
("Helper functions for " ++ showPretty (map IR.funcDeclName decls))
: helperDecls' ++ mainDecls')
convertRecFuncDeclsWithHelpers'
:: [IR.FuncDecl] -> Converter ([Coq.Sentence], [Coq.Sentence])
convertRecFuncDeclsWithHelpers' decls = do
decArgs <- identifyDecArgs decls
(helperDecls, mainDecls) <- mapAndUnzipM (uncurry transformRecFuncDecl)
(zip decls decArgs)
helperDecls'
<- forM (concat helperDecls) $ \(helperDecl, decArgIndex) -> localEnv $ do
inlinedHelperDecl <- inlineFuncDecls mainDecls helperDecl
let eliminatedHelperDecl = eliminateAliases inlinedHelperDecl decArgIndex
convertRecHelperFuncDecl eliminatedHelperDecl decArgIndex
mainDecls' <- convertNonRecFuncDecls mainDecls
return
( [Coq.FixpointSentence (Coq.Fixpoint (NonEmpty.fromList helperDecls') [])]
, mainDecls'
)
transformRecFuncDecl :: IR.FuncDecl
-> DecArgIndex
-> Converter ([(IR.FuncDecl, DecArgIndex)], IR.FuncDecl)
transformRecFuncDecl
(IR.FuncDecl srcSpan declIdent typeArgs args maybeRetType expr) decArgIndex
= do
(helperDecls, helperApps) <- mapAndUnzipM generateHelperDecl caseExprsPos
let mainExpr = replaceSubterms' expr (zip caseExprsPos helperApps)
mainDecl
= IR.FuncDecl srcSpan declIdent typeArgs args maybeRetType mainExpr
return (helperDecls, mainDecl)
where
name :: IR.QName
name = IR.declIdentName declIdent
argNames :: [IR.QName]
argNames = map IR.varPatQName args
decArg :: IR.QName
decArg = argNames !! decArgIndex
decArgAliasesAt :: Pos -> Set IR.QName
decArgAliasesAt p = Map.keysSet (Map.filter (== 0) (depthMapAt p expr decArg))
caseExprsPos :: [Pos]
caseExprsPos = let ps = map snd (findSubtermWithPos isCaseExpr expr)
in [p | p <- ps, not (any (below p) (delete p ps))]
isCaseExpr :: IR.Expr -> Pos -> Bool
isCaseExpr (IR.Case _ (IR.Var _ varName _) _ _) pos
= varName `Set.member` decArgAliasesAt pos
isCaseExpr _ _ = False
generateHelperDecl :: Pos -> Converter ((IR.FuncDecl, DecArgIndex), IR.Expr)
generateHelperDecl caseExprPos = do
helperName <- freshHaskellQName (IR.declIdentName declIdent)
let helperDeclIdent = declIdent { IR.declIdentName = helperName }
let helperTypeArgs = typeArgs
let aliasSubst = composeSubsts
[mkVarSubst alias decArg
| alias <- Set.toList (decArgAliasesAt caseExprPos)
]
caseExpr = selectSubterm' expr caseExprPos
caseExpr' = applySubst aliasSubst caseExpr
let boundVarTypeMap = boundVarsWithTypeAt expr caseExprPos
boundVars
= Map.keysSet boundVarTypeMap `Set.union` Set.fromList argNames
usedVars = freeVarSet caseExpr'
helperArgNames = Set.toList (usedVars `Set.intersection` boundVars)
let argTypes = map IR.varPatType args
argTypeMap = Map.fromList (zip argNames argTypes)
helperArgTypeMap = boundVarTypeMap `Map.union` argTypeMap
helperArgTypes = map (join . (`Map.lookup` helperArgTypeMap))
helperArgNames
argStrict = map IR.varPatIsStrict args
argStrictMap = Map.fromList (zip argNames argStrict)
helperArgStrict = map
((== decArg) .||. ((Just True ==) . (`Map.lookup` argStrictMap)))
helperArgNames
helperArgs = zipWith3
(IR.VarPat NoSrcSpan . fromJust . IR.identFromQName) helperArgNames
helperArgTypes helperArgStrict
helperReturnType = IR.exprType caseExpr'
helperType = IR.funcType NoSrcSpan <$> sequence helperArgTypes
<*> helperReturnType
freeArgsNeeded <- inEnv $ needsFreeArgs name
encEffects <- inEnv $ encapsulatesEffects name
effects <- inEnv $ lookupEffects name
_entry <- renameAndAddEntry
$ FuncEntry
{ entrySrcSpan = NoSrcSpan
, entryArity = length helperArgTypes
, entryTypeArgs = map IR.typeVarDeclIdent helperTypeArgs
, entryArgTypes = map fromJust helperArgTypes
, entryStrictArgs = map IR.varPatIsStrict helperArgs
, entryReturnType = fromJust helperReturnType
, entryNeedsFreeArgs = freeArgsNeeded
, entryEncapsulatesEffects = encEffects
, entryEffects = effects
, entryName = helperName
, entryIdent = undefined
, entryAgdaIdent = undefined
}
let decArgIndex' = fromJust $ elemIndex decArg helperArgNames
let helperTypeArgs' = map IR.typeVarDeclToType helperTypeArgs
helperAppType = IR.TypeScheme NoSrcSpan [] <$> helperType
helperDecl = IR.FuncDecl srcSpan helperDeclIdent helperTypeArgs
helperArgs helperReturnType caseExpr'
helperApp = IR.app NoSrcSpan
(IR.visibleTypeApp NoSrcSpan
(IR.Var NoSrcSpan helperName helperAppType) helperTypeArgs')
(map IR.varPatToExpr helperArgs)
let scrutinee = IR.caseExprScrutinee caseExpr
helperApp' = applySubst (singleSubst decArg scrutinee) helperApp
return ((helperDecl, decArgIndex'), helperApp')
eliminateAliases :: IR.FuncDecl -> DecArgIndex -> IR.FuncDecl
eliminateAliases helperDecl decArgIndex
= let decArg = IR.varPatQName (IR.funcDeclArgs helperDecl !! decArgIndex)
in helperDecl { IR.funcDeclRhs = eliminateAliases' (initDepthMap decArg)
(IR.funcDeclRhs helperDecl)
}
eliminateAliases' :: DepthMap -> IR.Expr -> IR.Expr
eliminateAliases' depthMap expr = case expr of
(IR.Let srcSpan binds inExpr exprType) ->
let (eliminatedBinds, perservedBinds) = partition shouldEliminate binds
names = map (IR.varPatQName . IR.bindVarPat) eliminatedBinds
exprs = map IR.bindExpr eliminatedBinds
subst = composeSubsts (zipWith singleSubst names exprs)
binds' = map (applySubst subst) perservedBinds
inExpr' = applySubst subst inExpr
letExpr' = IR.Let srcSpan binds' inExpr' exprType
in eliminateInChildren letExpr'
_ -> eliminateInChildren expr
where
shouldEliminate :: IR.Bind -> Bool
shouldEliminate = isJust . flip lookupDepth depthMap . IR.bindExpr
eliminateInChildren :: IR.Expr -> IR.Expr
eliminateInChildren expr'
= let children' = mapChildrenWithDepthMaps eliminateAliases' depthMap expr'
in fromJust (replaceChildTerms expr' children')
convertRecHelperFuncDecl :: IR.FuncDecl -> DecArgIndex -> Converter Coq.FixBody
convertRecHelperFuncDecl helperDecl decArgIndex = localEnv $ do
(qualid, binders, returnType') <- convertFuncHead helperDecl
rhs' <- convertExpr (IR.funcDeclRhs helperDecl)
Just decArg' <- inEnv
$ lookupIdent IR.ValueScope
$ IR.varPatQName
$ IR.funcDeclArgs helperDecl !! decArgIndex
return (Coq.FixBody qualid (NonEmpty.fromList binders)
(Just (Coq.StructOrder decArg')) returnType' rhs')