module FreeC.Frontend.Haskell.Simplifier
( Simplifier
, simplifyModuleWithComments
, simplifyModuleHeadWithComments
, simplifyModuleBody
, extractModName
, simplifyType
, simplifyTypeScheme
, simplifyExpr
) where
import Control.Monad ( unless, when )
import Control.Monad.Extra ( mapMaybeM )
import Data.Composition ( (.:) )
import Data.Maybe ( fromJust, fromMaybe, isJust )
import qualified Language.Haskell.Exts.Syntax as HSE
import FreeC.Environment.Fresh
import FreeC.Frontend.IR.PragmaParser
import qualified FreeC.IR.Base.Prelude as IR.Prelude
import FreeC.IR.Reference ( freeTypeVars )
import FreeC.IR.SrcSpan
import FreeC.IR.Subterm ( findFirstSubterm )
import qualified FreeC.IR.Syntax as IR
import FreeC.Monad.Converter
import FreeC.Monad.Reporter
type Simplifier = Converter
notSupported :: HSE.Annotated a
=> String
-> a SrcSpan
-> Simplifier r
notSupported feature = usageError (feature ++ " are not supported!")
skipNotSupported
:: HSE.Annotated a
=> String
-> a SrcSpan
-> Simplifier ()
skipNotSupported feature = skipNotSupported' feature "will be skipped"
skipNotSupported'
:: HSE.Annotated a
=> String
-> String
-> a SrcSpan
-> Simplifier ()
skipNotSupported' feature strategy node = report
$ Message (HSE.ann node) Warning
(feature ++ " are not supported and " ++ strategy ++ "!")
experimentallySupported
:: HSE.Annotated a => String -> a SrcSpan -> Simplifier r
experimentallySupported feature = usageError
(feature
++ " are not supported!\n"
++ "Add the `--transform-pattern-matching` command line option to enable "
++ feature)
expected :: HSE.Annotated a
=> String
-> a SrcSpan
-> Simplifier r
expected description = usageError ("Expected " ++ description ++ ".")
usageError :: HSE.Annotated a
=> String
-> a SrcSpan
-> Simplifier r
usageError message node = reportFatal $ Message (HSE.ann node) Error message
warnIf :: HSE.Annotated a
=> Bool
-> String
-> a SrcSpan
-> Simplifier ()
warnIf cond msg node = when cond (report $ Message (HSE.ann node) Warning msg)
simplifyModuleWithComments
:: HSE.Module SrcSpan -> [IR.Comment] -> Simplifier IR.Module
simplifyModuleWithComments inputModule comments = do
inputModule' <- simplifyModuleHeadWithComments inputModule comments
simplifyModuleBody inputModule'
simplifyModuleHeadWithComments :: HSE.Module SrcSpan
-> [IR.Comment]
-> Simplifier (IR.ModuleOf (HSE.Module SrcSpan))
simplifyModuleHeadWithComments
inputModule@(HSE.Module srcSpan _ pragmas imports _) comments = do
unless (null pragmas) $ skipNotSupported "Module pragmas" (head pragmas)
modName <- extractModName inputModule
imports' <- mapM simplifyImport imports
customPragmas <- liftReporter $ parseCustomPragmas comments
return IR.ModuleOf { IR.modSrcSpan = srcSpan
, IR.modName = modName
, IR.modImports = imports'
, IR.modPragmas = customPragmas
, IR.modContents = inputModule
}
simplifyModuleHeadWithComments modDecl _ = notSupported "XML modules" modDecl
simplifyModuleBody :: IR.ModuleOf (HSE.Module SrcSpan) -> Simplifier IR.Module
simplifyModuleBody inputModule = do
let HSE.Module _ _ _ _ decls = IR.modContents inputModule
decls' <- mapMaybeM simplifyDecl decls
return inputModule { IR.modContents = decls' }
extractModName :: HSE.Module SrcSpan -> Simplifier IR.ModName
extractModName (HSE.Module _ modHead _ _ _) = do
maybeModName <- mapM simplifyModuleHead modHead
return (fromMaybe "Main" maybeModName)
extractModName modDecl = notSupported "XML modules" modDecl
simplifyModuleHead :: HSE.ModuleHead SrcSpan -> Simplifier IR.ModName
simplifyModuleHead (HSE.ModuleHead _ (HSE.ModuleName _ modName) _ exports) = do
warnIf (isJust exports) "Ignoring export list." (fromJust exports)
return modName
simplifyImport :: HSE.ImportDecl SrcSpan -> Simplifier IR.ImportDecl
simplifyImport decl
| HSE.importQualified decl = notSupported "Qualified imports" decl
| HSE.importSrc decl = notSupported "Mutually recursive modules" decl
| HSE.importSafe decl = notSupported "Safe imports" decl
| isJust (HSE.importPkg decl)
= notSupported "Imports with explicit package names" decl
| isJust (HSE.importAs decl) = notSupported "Imports with aliases" decl
| isJust (HSE.importSpecs decl) = do
skipNotSupported' "Import specifications" "everything will be imported"
(fromJust (HSE.importSpecs decl))
simplifyImport decl { HSE.importSpecs = Nothing }
| otherwise = case HSE.importModule decl of
HSE.ModuleName srcSpan modName -> return (IR.ImportDecl srcSpan modName)
simplifyDecl :: HSE.Decl SrcSpan -> Simplifier (Maybe IR.TopLevelDecl)
simplifyDecl (HSE.TypeDecl srcSpan declHead typeExpr) = do
(declIdent, typeArgs) <- simplifyDeclHead declHead
typeExpr' <- simplifyType typeExpr
return (Just (IR.TopLevelTypeDecl
(IR.TypeSynDecl srcSpan declIdent typeArgs typeExpr')))
simplifyDecl
(HSE.DataDecl srcSpan (HSE.DataType _) Nothing declHead conDecls []) = do
(declIdent, typeArgs) <- simplifyDeclHead declHead
conDecls' <- mapM simplifyConDecl conDecls
return (Just (IR.TopLevelTypeDecl
(IR.DataDecl srcSpan declIdent typeArgs conDecls')))
simplifyDecl decl@(HSE.DataDecl _ (HSE.NewType _) _ _ _ _)
= notSupported "Newtype declarations" decl
simplifyDecl (HSE.DataDecl srcSpan dataType (Just context) declHead conDecls
derivingClauses) = do
skipNotSupported "Type class contexts" context
simplifyDecl
(HSE.DataDecl srcSpan dataType Nothing declHead conDecls derivingClauses)
simplifyDecl (HSE.DataDecl srcSpan dataType Nothing declHead conDecls
(derivingDecl : _)) = do
skipNotSupported "Deriving clauses" derivingDecl
simplifyDecl (HSE.DataDecl srcSpan dataType Nothing declHead conDecls [])
simplifyDecl (HSE.FunBind _ [match]) = do
funcDecl <- simplifyFuncDecl match
return (Just (IR.TopLevelFuncDecl funcDecl))
simplifyDecl decl@(HSE.FunBind _ _)
= experimentallySupported "Function declarations with more than one rule" decl
simplifyDecl (HSE.PatBind srcSpan (HSE.PVar _ declName) (HSE.UnGuardedRhs _ rhs)
Nothing) = do
declIdent <- simplifyFuncDeclName declName
rhs' <- simplifyExpr rhs
return (Just (IR.TopLevelFuncDecl
(IR.FuncDecl srcSpan declIdent [] [] Nothing rhs')))
simplifyDecl (HSE.PatBind _ (HSE.PVar _ _) rhss@(HSE.GuardedRhss _ _) _)
= experimentallySupported "Guards" rhss
simplifyDecl (HSE.PatBind _ (HSE.PVar _ _) _ (Just binds))
= notSupported "Local declarations" binds
simplifyDecl decl@(HSE.PatBind _ _ _ _)
= notSupported "Pattern-bindings other than 0-ary function declarations" decl
simplifyDecl (HSE.TypeSig srcSpan names typeExpr) = do
names' <- mapM simplifyFuncDeclName names
typeScheme' <- simplifyTypeScheme typeExpr
return (Just (IR.TopLevelTypeSig (IR.TypeSig srcSpan names' typeScheme')))
simplifyDecl (HSE.InfixDecl _ _ _ _) = return Nothing
simplifyDecl decl@(HSE.RulePragmaDecl _ _) = do
skipNotSupported "RULES pragmas" decl
return Nothing
simplifyDecl decl@(HSE.DeprPragmaDecl _ _) = do
skipNotSupported "DEPRECATED pragmas" decl
return Nothing
simplifyDecl decl@(HSE.WarnPragmaDecl _ _) = do
skipNotSupported "WARNING pragmas" decl
return Nothing
simplifyDecl decl@(HSE.InlineSig _ _ _ _) = do
skipNotSupported "INLINE pragmas" decl
return Nothing
simplifyDecl decl@(HSE.InlineConlikeSig _ _ _) = do
skipNotSupported "INLINE CONLIKE pragmas" decl
return Nothing
simplifyDecl decl@(HSE.SpecSig _ _ _ _) = do
skipNotSupported "SPECIALISE pragma" decl
return Nothing
simplifyDecl decl@(HSE.SpecInlineSig _ _ _ _ _) = do
skipNotSupported "SPECIALISE INLINE pragmas" decl
return Nothing
simplifyDecl decl@(HSE.InstSig _ _) = do
skipNotSupported "SPECIALISE instance pragmas" decl
return Nothing
simplifyDecl decl@(HSE.AnnPragma _ _) = do
skipNotSupported "ANN pragmas" decl
return Nothing
simplifyDecl decl@(HSE.MinimalPragma _ _) = do
skipNotSupported "MINIMAL pragmas" decl
return Nothing
simplifyDecl decl@(HSE.CompletePragma _ _ _) = do
skipNotSupported "COMPLETE pragma" decl
return Nothing
simplifyDecl decl@(HSE.TypeFamDecl _ _ _ _) = notSupported "Type families" decl
simplifyDecl decl@(HSE.ClosedTypeFamDecl _ _ _ _ _)
= notSupported "Type families" decl
simplifyDecl decl@(HSE.DataFamDecl _ _ _ _) = notSupported "Type families" decl
simplifyDecl decl@(HSE.TypeInsDecl _ _ _) = notSupported "Type families" decl
simplifyDecl decl@(HSE.DataInsDecl _ _ _ _ _)
= notSupported "Type families" decl
simplifyDecl decl@(HSE.GDataDecl _ _ _ _ _ _ _)
= notSupported "GADT style declarations" decl
simplifyDecl decl@(HSE.GDataInsDecl _ _ _ _ _ _)
= notSupported "GADT style declarations" decl
simplifyDecl decl@(HSE.ClassDecl _ _ _ _ _) = notSupported "Type classes" decl
simplifyDecl decl@(HSE.InstDecl _ _ _ _) = do
skipNotSupported "Instance declarations" decl
return Nothing
simplifyDecl decl@(HSE.DerivDecl _ _ _ _) = do
skipNotSupported "Deriving declarations" decl
return Nothing
simplifyDecl decl@(HSE.DefaultDecl _ _) = notSupported "Type classes" decl
simplifyDecl decl@(HSE.SpliceDecl _ _) = notSupported "Template Haskell" decl
simplifyDecl decl@(HSE.TSpliceDecl _ _) = notSupported "Template Haskell" decl
simplifyDecl decl@(HSE.PatSynSig _ _ _ _ _ _ _)
= notSupported "Pattern synonyms" decl
simplifyDecl decl@(HSE.PatSyn _ _ _ _) = notSupported "Pattern synonyms" decl
simplifyDecl decl@(HSE.ForImp _ _ _ _ _ _) = notSupported "Foreign imports" decl
simplifyDecl decl@(HSE.ForExp _ _ _ _ _) = notSupported "Foreign exports" decl
simplifyDecl decl@(HSE.RoleAnnotDecl _ _ _)
= notSupported "Role annotations" decl
simplifyDeclHead
:: HSE.DeclHead SrcSpan -> Simplifier (IR.DeclIdent, [IR.TypeVarDecl])
simplifyDeclHead (HSE.DHead _ declName) = do
declIdent <- simplifyDeclName declName
return (declIdent, [])
simplifyDeclHead (HSE.DHParen _ declHead)
= simplifyDeclHead declHead
simplifyDeclHead (HSE.DHApp _ declHead typeVarBind) = do
(declIdent, typeArgs) <- simplifyDeclHead declHead
typeArg <- simplifyTypeVarBind typeVarBind
return (declIdent, typeArgs ++ [typeArg])
simplifyDeclHead (HSE.DHInfix _ typeVarBind declName) = do
typeArg <- simplifyTypeVarBind typeVarBind
declIdent <- simplifyDeclName declName
return (declIdent, [typeArg])
simplifyDeclName :: HSE.Name SrcSpan -> Simplifier IR.DeclIdent
simplifyDeclName (HSE.Ident srcSpan ident) = return
(IR.DeclIdent srcSpan (IR.UnQual (IR.Ident ident)))
simplifyDeclName sym@(HSE.Symbol _ _) = notSupported "Type operators" sym
simplifyTypeVarBind :: HSE.TyVarBind SrcSpan -> Simplifier IR.TypeVarDecl
simplifyTypeVarBind (HSE.UnkindedVar srcSpan (HSE.Ident _ ident)) = return
(IR.TypeVarDecl srcSpan ident)
simplifyTypeVarBind typeVarBind@(HSE.UnkindedVar _ (HSE.Symbol _ _))
= notSupported "Type operators" typeVarBind
simplifyTypeVarBind typeVarBind@(HSE.KindedVar _ _ _)
= notSupported "Kind annotations" typeVarBind
simplifyConDecl :: HSE.QualConDecl SrcSpan -> Simplifier IR.ConDecl
simplifyConDecl (HSE.QualConDecl _ Nothing Nothing conDecl)
= simplifyConDecl' conDecl
simplifyConDecl conDecl@(HSE.QualConDecl _ (Just _) _ _)
= notSupported "Existential quantifications" conDecl
simplifyConDecl conDecl@(HSE.QualConDecl _ _ (Just _) _)
= notSupported "Type classes" conDecl
simplifyConDecl' :: HSE.ConDecl SrcSpan -> Simplifier IR.ConDecl
simplifyConDecl' (HSE.ConDecl srcSpan conName args) = do
conIdent <- simplifyConDeclName conName
args' <- mapM simplifyType args
return (IR.ConDecl srcSpan conIdent args')
simplifyConDecl' (HSE.InfixConDecl pos t1 conName t2) = simplifyConDecl'
(HSE.ConDecl pos conName [t1, t2])
simplifyConDecl' conDecl@(HSE.RecDecl _ _ _)
= notSupported "Record constructors" conDecl
simplifyConDeclName :: HSE.Name SrcSpan -> Simplifier IR.DeclIdent
simplifyConDeclName (HSE.Ident srcSpan ident) = return
(IR.DeclIdent srcSpan (IR.UnQual (IR.Ident ident)))
simplifyConDeclName sym@(HSE.Symbol _ _)
= notSupported "Constructor operator declarations" sym
simplifyFuncDecl :: HSE.Match SrcSpan -> Simplifier IR.FuncDecl
simplifyFuncDecl (HSE.Match srcSpan declName args (HSE.UnGuardedRhs _ rhs)
Nothing) = do
declIdent <- simplifyFuncDeclName declName
args' <- mapM simplifyVarPat args
rhs' <- simplifyExpr rhs
return (IR.FuncDecl srcSpan declIdent [] args' Nothing rhs')
simplifyFuncDecl (HSE.Match _ _ _ rhss@(HSE.GuardedRhss _ _) _)
= experimentallySupported "Guards" rhss
simplifyFuncDecl (HSE.Match _ _ _ _ (Just binds))
= notSupported "Local declarations" binds
simplifyFuncDecl (HSE.InfixMatch pos arg declName args rhs binds)
= simplifyFuncDecl (HSE.Match pos declName (arg : args) rhs binds)
simplifyFuncDeclName :: HSE.Name SrcSpan -> Simplifier IR.DeclIdent
simplifyFuncDeclName (HSE.Ident srcSpan ident) = return
(IR.DeclIdent srcSpan (IR.UnQual (IR.Ident ident)))
simplifyFuncDeclName sym@(HSE.Symbol _ _)
= notSupported "Operator declarations" sym
simplifyTypeScheme :: HSE.Type SrcSpan -> Simplifier IR.TypeScheme
simplifyTypeScheme (HSE.TyForall srcSpan (Just binds) Nothing typeExpr) = do
typeArgs <- mapM simplifyTypeVarBind binds
typeExpr' <- simplifyType typeExpr
return (IR.TypeScheme srcSpan typeArgs typeExpr')
simplifyTypeScheme typeExpr = do
typeExpr' <- simplifyType typeExpr
let srcSpan = IR.typeSrcSpan typeExpr'
typeArgIdents = freeTypeVars typeExpr'
typeArgSrcSpans = map (findTypeArgSrcSpan typeExpr') typeArgIdents
typeArgs = zipWith IR.TypeVarDecl typeArgSrcSpans typeArgIdents
return (IR.TypeScheme srcSpan typeArgs typeExpr')
where
findTypeArgSrcSpan :: IR.Type -> IR.TypeVarIdent -> SrcSpan
findTypeArgSrcSpan = fromMaybe NoSrcSpan .: flip findTypeArgSrcSpan'
findTypeArgSrcSpan' :: IR.TypeVarIdent -> IR.Type -> Maybe SrcSpan
findTypeArgSrcSpan' = fmap IR.typeSrcSpan .: findFirstSubterm . isTypeVar
isTypeVar :: IR.TypeVarIdent -> IR.Type -> Bool
isTypeVar typeVarIdent (IR.TypeVar _ typeVarIdent')
= typeVarIdent == typeVarIdent'
isTypeVar _ _ = False
simplifyType :: HSE.Type SrcSpan -> Simplifier IR.Type
simplifyType (HSE.TyFun srcSpan t1 t2) = do
t1' <- simplifyType t1
t2' <- simplifyType t2
return (IR.FuncType srcSpan t1' t2')
simplifyType (HSE.TyTuple srcSpan HSE.Boxed ts) = do
let n = length ts
ts' <- mapM simplifyType ts
return (IR.typeConApp srcSpan (IR.Prelude.tupleTypeConName n) ts')
simplifyType (HSE.TyList srcSpan t) = do
t' <- simplifyType t
return (IR.typeConApp srcSpan IR.Prelude.listTypeConName [t'])
simplifyType (HSE.TyApp srcSpan t1 t2) = do
t1' <- simplifyType t1
t2' <- simplifyType t2
return (IR.TypeApp srcSpan t1' t2')
simplifyType (HSE.TyVar srcSpan (HSE.Ident _ ident)) = return
(IR.TypeVar srcSpan ident)
simplifyType (HSE.TyCon srcSpan name) = do
name' <- simplifyTypeConName name
return (IR.TypeCon srcSpan name')
simplifyType (HSE.TyParen _ t) = simplifyType t
simplifyType (HSE.TyForall _ Nothing (Just context) t) = do
skipNotSupported "Type class contexts" context
simplifyType t
simplifyType ty@(HSE.TyForall _ _ _ _)
= notSupported "Explicit type variable quantifications" ty
simplifyType ty@(HSE.TyTuple _ HSE.Unboxed _) = notSupported "Unboxed tuples" ty
simplifyType ty@(HSE.TyUnboxedSum _ _) = notSupported "Unboxed sums" ty
simplifyType ty@(HSE.TyParArray _ _) = notSupported "Parallel arrays" ty
simplifyType ty@(HSE.TyKind _ _ _)
= notSupported "Types with explicit kind signatures" ty
simplifyType ty@(HSE.TyStar _) = notSupported "Kinds" ty
simplifyType ty@(HSE.TyVar _ (HSE.Symbol _ _))
= notSupported "Type operators" ty
simplifyType ty@(HSE.TyPromoted _ _) = notSupported "Type operators" ty
simplifyType ty@(HSE.TyInfix _ _ _ _) = notSupported "Type operators" ty
simplifyType ty@(HSE.TyEquals _ _ _)
= notSupported "Type equality predicates" ty
simplifyType ty@(HSE.TySplice _ _) = notSupported "Template Haskell" ty
simplifyType ty@(HSE.TyBang _ _ _ _) = notSupported "Strictness annotations" ty
simplifyType ty@(HSE.TyWildCard _ _) = notSupported "Type wildcards" ty
simplifyType ty@(HSE.TyQuasiQuote _ _ _)
= notSupported "Quasiquotation types" ty
simplifyTypeConName :: HSE.QName SrcSpan -> Simplifier IR.TypeConName
simplifyTypeConName (HSE.UnQual _ (HSE.Ident _ ident)) = return
(IR.UnQual (IR.Ident ident))
simplifyTypeConName (HSE.Qual _ (HSE.ModuleName _ modName) (HSE.Ident _ ident))
= return (IR.Qual modName (IR.Ident ident))
simplifyTypeConName (HSE.Special _ (HSE.UnitCon _))
= return IR.Prelude.unitTypeConName
simplifyTypeConName (HSE.Special _ (HSE.ListCon _))
= return IR.Prelude.listTypeConName
simplifyTypeConName (HSE.Special _ (HSE.TupleCon _ HSE.Boxed n)) = return
(IR.Prelude.tupleTypeConName n)
simplifyTypeConName name@(HSE.UnQual _ (HSE.Symbol _ _))
= notSupported "Type operators" name
simplifyTypeConName name@(HSE.Qual _ _ (HSE.Symbol _ _))
= notSupported "Type operators" name
simplifyTypeConName name@(HSE.Special _ (HSE.FunCon _))
= notSupported "Function type constructors" name
simplifyTypeConName name@(HSE.Special _ (HSE.TupleCon _ HSE.Unboxed _))
= notSupported "Unboxed tuples" name
simplifyTypeConName name@(HSE.Special _ (HSE.UnboxedSingleCon _))
= notSupported "Unboxed tuples" name
simplifyTypeConName name@(HSE.Special _ (HSE.ExprHole _))
= notSupported "Expression holes" name
simplifyTypeConName name@(HSE.Special _ (HSE.Cons _)) = usageError
"The data constructor (:) cannot be used as a type constructor!" name
simplifyExpr :: HSE.Exp SrcSpan -> Simplifier IR.Expr
simplifyExpr (HSE.Var srcSpan (HSE.UnQual _ (HSE.Ident _ "undefined"))) = return
(IR.Undefined srcSpan Nothing)
simplifyExpr (HSE.App srcSpan (HSE.Var _ (HSE.UnQual _ (HSE.Ident _ "error")))
msgArg) = case msgArg of
(HSE.Lit _ (HSE.String _ msg _)) -> return (IR.ErrorExpr srcSpan msg Nothing)
_ -> notSupported "Non-literal error messages" msgArg
simplifyExpr expr@(HSE.Var _ (HSE.UnQual _ (HSE.Ident _ "error")))
= usageError "The function 'error' must be applied immediately." expr
simplifyExpr
(HSE.App srcSpan (HSE.App _ (HSE.Var _ (HSE.UnQual _ (HSE.Ident _ "trace")))
msgArg) arg) = case msgArg of
(HSE.Lit _ (HSE.String _ msg _)) -> do
arg' <- simplifyExpr arg
return (IR.Trace srcSpan msg arg' Nothing)
_ -> notSupported "Non-literal error messages" msgArg
simplifyExpr expr@(HSE.Var _ (HSE.UnQual _ (HSE.Ident _ "trace")))
= usageError "The function 'trace' must be applied twice immediately." expr
simplifyExpr (HSE.Paren _ expr) = simplifyExpr expr
simplifyExpr (HSE.Var srcSpan name) = do
name' <- simplifyVarName name
return (IR.Var srcSpan name' Nothing)
simplifyExpr (HSE.Con srcSpan name) = do
name' <- simplifyConName name
return (IR.Con srcSpan name' Nothing)
simplifyExpr (HSE.Lit srcSpan (HSE.Int _ value _)) = return
(IR.IntLiteral srcSpan value Nothing)
simplifyExpr (HSE.Tuple srcSpan HSE.Boxed es) = do
let n = length es
es' <- mapM simplifyExpr es
return (IR.conApp srcSpan (IR.Prelude.tupleConName n) es')
simplifyExpr (HSE.List srcSpan exprs) = do
let nil = IR.Con srcSpan IR.Prelude.nilConName Nothing
cons = IR.Con srcSpan IR.Prelude.consConName Nothing
exprs' <- mapM simplifyExpr exprs
return (foldr (IR.untypedApp srcSpan . IR.untypedApp srcSpan cons) nil exprs')
simplifyExpr (HSE.App srcSpan e1 e2) = do
e1' <- simplifyExpr e1
e2' <- simplifyExpr e2
return (IR.App srcSpan e1' e2' Nothing)
simplifyExpr (HSE.InfixApp srcSpan e1 op e2) = do
e1' <- simplifyExpr e1
op' <- simplifyOp op
e2' <- simplifyExpr e2
return (IR.app srcSpan op' [e1', e2'])
simplifyExpr (HSE.LeftSection srcSpan e1 op) = do
e1' <- simplifyExpr e1
op' <- simplifyOp op
return (IR.App srcSpan op' e1' Nothing)
simplifyExpr (HSE.RightSection srcSpan op e2) = do
x <- freshHaskellIdent freshArgPrefix
op' <- simplifyOp op
e2' <- simplifyExpr e2
let x' = IR.VarPat srcSpan x Nothing False
e1' = IR.Var srcSpan (IR.UnQual (IR.Ident x)) Nothing
return (IR.Lambda srcSpan [x'] (IR.app srcSpan op' [e1', e2']) Nothing)
simplifyExpr (HSE.NegApp srcSpan expr) = do
expr' <- simplifyExpr expr
return (IR.varApp srcSpan IR.Prelude.negateOpName [expr'])
simplifyExpr (HSE.Lambda srcSpan args expr) = do
args' <- mapM simplifyVarPat args
expr' <- simplifyExpr expr
return (IR.Lambda srcSpan args' expr' Nothing)
simplifyExpr (HSE.If srcSpan e1 e2 e3) = do
e1' <- simplifyExpr e1
e2' <- simplifyExpr e2
e3' <- simplifyExpr e3
return (IR.If srcSpan e1' e2' e3' Nothing)
simplifyExpr (HSE.Case caseSrcSpan scrutinee
[ HSE.Alt altSrcSpan (HSE.PVar patSrcSpan (HSE.Ident _ ident))
(HSE.UnGuardedRhs _ expr) Nothing
]) = do
scrutinee' <- simplifyExpr scrutinee
expr' <- simplifyExpr expr
let pat' = IR.VarPat patSrcSpan ident Nothing False
lambda' = IR.Lambda altSrcSpan [pat'] expr' Nothing
return (IR.App caseSrcSpan lambda' scrutinee' Nothing)
simplifyExpr (HSE.Case srcSpan expr alts) = do
expr' <- simplifyExpr expr
alts' <- mapM simplifyAlt alts
return (IR.Case srcSpan expr' alts' Nothing)
simplifyExpr (HSE.Let srcSpan binds expr) = do
expr' <- simplifyExpr expr
binds' <- simplifyBinds binds
return (IR.Let srcSpan binds' expr' Nothing)
simplifyExpr (HSE.ExpTypeSig srcSpan expr typeExpr) = do
expr' <- simplifyExpr expr
case IR.exprTypeScheme expr' of
Nothing -> do
typeScheme <- simplifyTypeScheme typeExpr
return expr' { IR.exprTypeScheme = Just typeScheme }
Just _ -> do
report
$ Message srcSpan Warning
$ "Type signature is redundant and will be ignored."
return expr'
simplifyExpr pragma@(HSE.CorePragma _ _ expr) = do
skipNotSupported "CORE pragmas" pragma
simplifyExpr expr
simplifyExpr pragma@(HSE.SCCPragma _ _ expr) = do
skipNotSupported "SCC pragmas" pragma
simplifyExpr expr
simplifyExpr pragma@(HSE.GenPragma _ _ _ _ expr) = do
skipNotSupported "GENERATED pragmas" pragma
simplifyExpr expr
simplifyExpr expr@(HSE.OverloadedLabel _ _)
= notSupported "Overloaded labels" expr
simplifyExpr expr@(HSE.IPVar _ _)
= notSupported "Implicit parameter variables" expr
simplifyExpr expr@(HSE.MultiIf _ _)
= notSupported "Multi-Way if expressions" expr
simplifyExpr expr@(HSE.Do _ _) = notSupported "do-expressions" expr
simplifyExpr expr@(HSE.MDo _ _) = notSupported "mdo-expressions" expr
simplifyExpr expr@(HSE.Tuple _ HSE.Unboxed _)
= notSupported "Unboxed tuples" expr
simplifyExpr expr@(HSE.UnboxedSum _ _ _ _) = notSupported "Unboxed sums" expr
simplifyExpr expr@(HSE.TupleSection _ _ _) = notSupported "Tuple sections" expr
simplifyExpr expr@(HSE.ParArray _ _) = notSupported "Parallel arrays" expr
simplifyExpr expr@(HSE.RecConstr _ _ _)
= notSupported "Record constructors" expr
simplifyExpr expr@(HSE.RecUpdate _ _ _) = notSupported "Record updates" expr
simplifyExpr expr@(HSE.EnumFrom _ _) = notSupported "Enumerations" expr
simplifyExpr expr@(HSE.EnumFromTo _ _ _) = notSupported "Enumerations" expr
simplifyExpr expr@(HSE.EnumFromThen _ _ _) = notSupported "Enumerations" expr
simplifyExpr expr@(HSE.EnumFromThenTo _ _ _ _)
= notSupported "Enumerations" expr
simplifyExpr expr@(HSE.ParArrayFromTo _ _ _)
= notSupported "Parallel arrays" expr
simplifyExpr expr@(HSE.ParArrayFromThenTo _ _ _ _)
= notSupported "Parallel arrays" expr
simplifyExpr expr@(HSE.ListComp _ _ _) = notSupported "List comprehensions" expr
simplifyExpr expr@(HSE.ParComp _ _ _)
= notSupported "Parallel list comprehensions" expr
simplifyExpr expr@(HSE.ParArrayComp _ _ _)
= notSupported "Parallel array comprehensions" expr
simplifyExpr expr@(HSE.VarQuote _ _) = notSupported "Template Haskell" expr
simplifyExpr expr@(HSE.TypQuote _ _) = notSupported "Template Haskell" expr
simplifyExpr expr@(HSE.BracketExp _ _) = notSupported "Template Haskell" expr
simplifyExpr expr@(HSE.SpliceExp _ _) = notSupported "Template Haskell" expr
simplifyExpr expr@(HSE.QuasiQuote _ _ _) = notSupported "Quasiquotation" expr
simplifyExpr expr@(HSE.TypeApp _ _)
= notSupported "Visible type applications" expr
simplifyExpr expr@(HSE.XTag _ _ _ _ _) = notSupported "XML elements" expr
simplifyExpr expr@(HSE.XETag _ _ _ _) = notSupported "XML elements" expr
simplifyExpr expr@(HSE.XPcdata _ _) = notSupported "XML elements" expr
simplifyExpr expr@(HSE.XExpTag _ _) = notSupported "XML elements" expr
simplifyExpr expr@(HSE.XChildTag _ _) = notSupported "XML elements" expr
simplifyExpr expr@(HSE.Proc _ _ _) = notSupported "Arrow expressions" expr
simplifyExpr expr@(HSE.LeftArrApp _ _ _) = notSupported "Arrow expressions" expr
simplifyExpr expr@(HSE.RightArrApp _ _ _)
= notSupported "Arrow expressions" expr
simplifyExpr expr@(HSE.LeftArrHighApp _ _ _)
= notSupported "Arrow expressions" expr
simplifyExpr expr@(HSE.RightArrHighApp _ _ _)
= notSupported "Arrow expressions" expr
simplifyExpr expr@(HSE.ArrOp _ _) = notSupported "Arrow control operators" expr
simplifyExpr expr@(HSE.LCase _ _) = notSupported "Lambda case expressions" expr
simplifyExpr expr@(HSE.Lit _ (HSE.Char _ _ _))
= notSupported "Character literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.String _ _ _))
= notSupported "String literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.Frac _ _ _))
= notSupported "Floating point literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.PrimInt _ _ _))
= notSupported "Unboxed integer literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.PrimWord _ _ _))
= notSupported "Unboxed word literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.PrimFloat _ _ _))
= notSupported "Unboxed float literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.PrimDouble _ _ _))
= notSupported "Unboxed double literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.PrimChar _ _ _))
= notSupported "Unboxed character literals" expr
simplifyExpr expr@(HSE.Lit _ (HSE.PrimString _ _ _))
= notSupported "Unboxed string literals" expr
simplifyOp :: HSE.QOp SrcSpan -> Simplifier IR.Expr
simplifyOp (HSE.QVarOp srcSpan name)
= IR.untypedVar srcSpan <$> simplifyVarName name
simplifyOp (HSE.QConOp srcSpan name)
= IR.untypedCon srcSpan <$> simplifyConName name
simplifyName :: HSE.Name SrcSpan -> Simplifier IR.Name
simplifyName (HSE.Ident _ ident) = return (IR.Ident ident)
simplifyName (HSE.Symbol _ sym) = return (IR.Symbol sym)
simplifyVarName :: HSE.QName SrcSpan -> Simplifier IR.VarName
simplifyVarName (HSE.UnQual _ name) = IR.UnQual <$> simplifyName name
simplifyVarName (HSE.Qual _ (HSE.ModuleName _ modName) name)
= IR.Qual modName <$> simplifyName name
simplifyVarName name@(HSE.Special _ _)
= usageError "Constructors cannot be used as variables!" name
simplifyConName :: HSE.QName SrcSpan -> Simplifier IR.ConName
simplifyConName (HSE.UnQual _ name) = IR.UnQual <$> simplifyName name
simplifyConName (HSE.Qual _ (HSE.ModuleName _ modName) name)
= IR.Qual modName <$> simplifyName name
simplifyConName (HSE.Special _ (HSE.UnitCon _)) = return IR.Prelude.unitConName
simplifyConName (HSE.Special _ (HSE.ListCon _)) = return IR.Prelude.nilConName
simplifyConName (HSE.Special _ (HSE.Cons _)) = return IR.Prelude.consConName
simplifyConName (HSE.Special _ (HSE.TupleCon _ HSE.Boxed n)) = return
(IR.Prelude.tupleConName n)
simplifyConName name@(HSE.Special _ (HSE.FunCon _))
= usageError "Function type constructor cannot be used as a constructor!" name
simplifyConName name@(HSE.Special _ (HSE.TupleCon _ HSE.Unboxed _))
= notSupported "Unboxed tuples" name
simplifyConName name@(HSE.Special _ (HSE.UnboxedSingleCon _))
= notSupported "Unboxed tuples" name
simplifyConName name@(HSE.Special _ (HSE.ExprHole _))
= notSupported "Expression holes" name
simplifyVarPat :: HSE.Pat SrcSpan -> Simplifier IR.VarPat
simplifyVarPat (HSE.PVar srcSpan (HSE.Ident _ ident)) = return
(IR.VarPat srcSpan ident Nothing False)
simplifyVarPat pat = expected "variable pattern" pat
simplifyConPat :: HSE.Pat SrcSpan -> Simplifier (IR.ConPat, [IR.VarPat])
simplifyConPat (HSE.PParen _ pat) = simplifyConPat pat
simplifyConPat (HSE.PApp _ name args) = do
name' <- simplifyConName name
vars <- mapM simplifyVarPat args
return (IR.ConPat (HSE.ann name) name', vars)
simplifyConPat (HSE.PInfixApp _ p1 name p2) = do
v1 <- simplifyVarPat p1
name' <- simplifyConName name
v2 <- simplifyVarPat p2
return (IR.ConPat (HSE.ann name) name', [v1, v2])
simplifyConPat (HSE.PTuple srcSpan HSE.Boxed ps) = do
let n = length ps
vs <- mapM simplifyVarPat ps
return (IR.ConPat srcSpan (IR.Prelude.tupleConName n), vs)
simplifyConPat pat@(HSE.PTuple _ HSE.Unboxed _)
= notSupported "Unboxed tuples" pat
simplifyConPat (HSE.PList srcSpan [])
= return (IR.ConPat srcSpan IR.Prelude.nilConName, [])
simplifyConPat pat@(HSE.PList _ _)
= experimentallySupported "List notation patterns" pat
simplifyConPat pat@(HSE.PRec _ _ _) = notSupported "Record constructors" pat
simplifyConPat pat = expected "constructor pattern" pat
simplifyAlt :: HSE.Alt SrcSpan -> Simplifier IR.Alt
simplifyAlt (HSE.Alt srcSpan pat (HSE.UnGuardedRhs _ expr) Nothing) = do
(con, vars) <- simplifyConPat pat
expr' <- simplifyExpr expr
return (IR.Alt srcSpan con vars expr')
simplifyAlt (HSE.Alt _ _ rhss@(HSE.GuardedRhss _ _) _)
= experimentallySupported "Guards" rhss
simplifyAlt (HSE.Alt _ _ _ (Just binds))
= notSupported "Local declarations" binds
simplifyBinds :: HSE.Binds SrcSpan -> Simplifier [IR.Bind]
simplifyBinds binds@(HSE.IPBinds _ _) = notSupported "Implicit parameters" binds
simplifyBinds (HSE.BDecls _ decls) = mapM simplifyBind decls
where
simplifyBind :: HSE.Decl SrcSpan -> Simplifier IR.Bind
simplifyBind (HSE.PatBind srcSpan varPat (HSE.UnGuardedRhs _ expr) Nothing)
= do
varPat' <- simplifyVarPat varPat
expr' <- simplifyExpr expr
return (IR.Bind srcSpan varPat' expr')
simplifyBind (HSE.PatBind _ _ rhss@(HSE.GuardedRhss _ _) _)
= experimentallySupported "Guards" rhss
simplifyBind (HSE.PatBind _ _ _ (Just binds))
= notSupported "Local declarations" binds
simplifyBind decl = expected "a variable pattern binding" decl