diff options
author | partain <unknown> | 1996-03-21 12:48:09 +0000 |
---|---|---|
committer | partain <unknown> | 1996-03-21 12:48:09 +0000 |
commit | 0596517a9b4b2b32e5d375a986351102ac4540fc (patch) | |
tree | 1d3cdb3153c68ffaeccde89070f0fca3f1af5d77 | |
parent | 6c381e873e222417d9a67aeec77b9555eca7b7a8 (diff) | |
download | haskell-0596517a9b4b2b32e5d375a986351102ac4540fc.tar.gz |
[project @ 1996-03-21 12:46:33 by partain]
Final compiler stuff before Sansom renamer 960321
55 files changed, 1088 insertions, 997 deletions
diff --git a/ghc/compiler/Jmakefile b/ghc/compiler/Jmakefile index 1d167587b4..aecfcbd1b0 100644 --- a/ghc/compiler/Jmakefile +++ b/ghc/compiler/Jmakefile @@ -216,8 +216,7 @@ stranal/WwLib.lhs \ stranal/WorkWrap.lhs \ \ profiling/SCCauto.lhs \ -profiling/SCCfinal.lhs \ -profiling/CostCentre.lhs +profiling/SCCfinal.lhs #if GhcWithDeforester != YES #define __omit_deforester_flag -DOMIT_DEFORESTER=1 @@ -924,7 +923,7 @@ compile(parser/U_ttype,hs,$(PARSER_HS_OPTS) '-#include"hspincl.h"') /* *** misc *************************************************** */ -DEPSRCS = $(ALLSRCS_LHS) $(ALLSRCS_HS) +DEPSRCS = $(ALLSRCS_LHS) $(ALLSRCS_HS) SIMPL_SRCS_LHS #if GhcWithHscBuiltViaC == NO MKDEPENDHS_OPTS= -I$(MAIN_INCLUDE_DIR) -I$(COMPINFO_DIR) -x HsVersions.h diff --git a/ghc/compiler/basicTypes/Id.lhs b/ghc/compiler/basicTypes/Id.lhs index ff7deabd8c..ec6367e937 100644 --- a/ghc/compiler/basicTypes/Id.lhs +++ b/ghc/compiler/basicTypes/Id.lhs @@ -110,12 +110,14 @@ import PprStyle import Pretty import SrcLoc ( mkBuiltinSrcLoc ) import TyCon ( TyCon, mkTupleTyCon, getTyConDataCons ) -import Type ( mkSigmaTy, mkTyVarTy, mkFunTys, mkDictTy, - applyTyCon, isPrimType, instantiateTy, - GenType, ThetaType(..), TauType(..), Type(..) ) -import TyVar ( GenTyVar, alphaTyVars ) +import Type ( mkSigmaTy, mkTyVarTys, mkFunTys, mkDictTy, + applyTyCon, isPrimType, instantiateTy, + tyVarsOfType, + GenType, ThetaType(..), TauType(..), Type(..) + ) +import TyVar ( GenTyVar, alphaTyVars, isEmptyTyVarSet ) import UniqFM -import UniqSet ( UniqSet(..) ) +import UniqSet -- practically all of it import Unique ( Unique, mkTupleDataConUnique, pprUnique, showUnique ) import Util ( mapAccumL, nOfThem, panic, pprPanic, assertPanic ) \end{code} @@ -480,7 +482,7 @@ toplevelishId (Id _ _ details _ _) chk (PreludeId _) = True chk (TopLevId _) = True -- NB: see notes chk (SuperDictSelId _ _) = True - chk (MethodSelId _ _) = True + chk (MethodSelId _ _) = True chk (DefaultMethodId _ _ _) = True chk (DictFunId _ _ _ _) = True chk (ConstMethodId _ _ _ _ _) = True @@ -501,7 +503,7 @@ idHasNoFreeTyVars (Id _ _ details _ info) chk (PreludeId _) = True chk (TopLevId _) = True chk (SuperDictSelId _ _) = True - chk (MethodSelId _ _) = True + chk (MethodSelId _ _) = True chk (DefaultMethodId _ _ _) = True chk (DictFunId _ _ _ _) = True chk (ConstMethodId _ _ _ _ _) = True @@ -814,10 +816,11 @@ externallyVisibleId id@(Id _ _ details _ _) \end{code} \begin{code} -{-LATER: idWantsToBeINLINEd :: Id -> Bool idWantsToBeINLINEd id + = panic "Id.idWantsToBeINLINEd" +{- LATER: = case (getIdUnfolding id) of IWantToBeINLINEd _ -> True _ -> False @@ -1176,11 +1179,14 @@ updateIdType (Id u _ info details) ty = Id u ty info details \end{code} \begin{code} -no_free_tvs ty = panic "Id:no_free_tvs" -- null (extractTyVarsFromTy ty) +type MyTy a b = GenType (GenTyVar a) b +type MyId a b = GenId (MyTy a b) + +no_free_tvs ty = isEmptyTyVarSet (tyVarsOfType ty) -- SysLocal: for an Id being created by the compiler out of thin air... -- UserLocal: an Id with a name the user might recognize... -mkSysLocal, mkUserLocal :: FAST_STRING -> Unique -> ty -> SrcLoc -> GenId ty +mkSysLocal, mkUserLocal :: FAST_STRING -> Unique -> MyTy a b -> SrcLoc -> MyId a b mkSysLocal str uniq ty loc = Id uniq ty (SysLocalId (mkShortName str loc) (no_free_tvs ty)) NoPragmaInfo noIdInfo @@ -1189,7 +1195,7 @@ mkUserLocal str uniq ty loc = Id uniq ty (LocalId (mkShortName str loc) (no_free_tvs ty)) NoPragmaInfo noIdInfo -- mkUserId builds a local or top-level Id, depending on the name given -mkUserId :: Name -> ty -> PragmaInfo -> GenId ty +mkUserId :: Name -> MyTy a b -> PragmaInfo -> MyId a b mkUserId (Short uniq short) ty pragma_info = Id uniq ty (LocalId short (no_free_tvs ty)) pragma_info noIdInfo mkUserId (ValName uniq full) ty pragma_info @@ -1342,7 +1348,7 @@ mkDataCon k n stricts tvs ctxt args_tys tycon type_of_constructor = mkSigmaTy tvs ctxt - (mkFunTys args_tys (applyTyCon tycon (map mkTyVarTy tvs))) + (mkFunTys args_tys (applyTyCon tycon (mkTyVarTys tvs))) datacon_info = noIdInfo `addInfo_UF` unfolding `addInfo` mkArityInfo arity @@ -1358,7 +1364,7 @@ mkDataCon k n stricts tvs ctxt args_tys tycon -- else -- do some business... let (tyvars, dict_vars, vars) = mk_uf_bits tvs ctxt args_tys tycon - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars in BIND (Con data_con tyvar_tys [VarArg v | v <- vars]) _TO_ plain_Con -> @@ -1406,7 +1412,7 @@ mkTupleCon arity (mkFunTys tyvar_tys (applyTyCon tycon tyvar_tys)) tycon = mkTupleTyCon arity tyvars = take arity alphaTyVars - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars tuplecon_info = noIdInfo `addInfo_UF` unfolding @@ -1421,7 +1427,7 @@ mkTupleCon arity -- else -- do some business... let (tyvars, dict_vars, vars) = mk_uf_bits arity - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars in BIND (Con data_con tyvar_tys [VarArg v | v <- vars]) _TO_ plain_Con -> @@ -1463,7 +1469,7 @@ getDataConSig (Id _ _ (TupleConId arity) _ _) = (tyvars, [], tyvar_tys, mkTupleTyCon arity) where tyvars = take arity alphaTyVars - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars \end{code} {- LATER @@ -1758,7 +1764,7 @@ is_prelude_core_ty :: Type -> Bool is_prelude_core_ty inst_ty = panic "Id.is_prelude_core_ty" {- LATER - = case maybeDataTyCon inst_ty of + = case maybeAppDataTyCon inst_ty of Just (tycon,_,_) -> fromPreludeCore tycon Nothing -> panic "Id: is_prelude_core_ty" -} @@ -2042,4 +2048,26 @@ modifyIdEnv env mangle_fn key \begin{code} type GenIdSet ty = UniqSet (GenId ty) type IdSet = UniqSet (GenId Type) + +emptyIdSet :: GenIdSet ty +intersectIdSets :: GenIdSet ty -> GenIdSet ty -> GenIdSet ty +unionIdSets :: GenIdSet ty -> GenIdSet ty -> GenIdSet ty +unionManyIdSets :: [GenIdSet ty] -> GenIdSet ty +idSetToList :: GenIdSet ty -> [GenId ty] +singletonIdSet :: GenId ty -> GenIdSet ty +elementOfIdSet :: GenId ty -> GenIdSet ty -> Bool +minusIdSet :: GenIdSet ty -> GenIdSet ty -> GenIdSet ty +isEmptyIdSet :: GenIdSet ty -> Bool +mkIdSet :: [GenId ty] -> GenIdSet ty + +emptyIdSet = emptyUniqSet +singletonIdSet = singletonUniqSet +intersectIdSets = intersectUniqSets +unionIdSets = unionUniqSets +unionManyIdSets = unionManyUniqSets +idSetToList = uniqSetToList +elementOfIdSet = elementOfUniqSet +minusIdSet = minusUniqSet +isEmptyIdSet = isEmptyUniqSet +mkIdSet = mkUniqSet \end{code} diff --git a/ghc/compiler/basicTypes/UniqSupply.lhs b/ghc/compiler/basicTypes/UniqSupply.lhs index 425e0459e1..81fec96096 100644 --- a/ghc/compiler/basicTypes/UniqSupply.lhs +++ b/ghc/compiler/basicTypes/UniqSupply.lhs @@ -14,7 +14,7 @@ module UniqSupply ( UniqSM(..), -- type: unique supply monad initUs, thenUs, returnUs, - mapUs, mapAndUnzipUs, + mapUs, mapAndUnzipUs, mapAndUnzip3Us, mkSplitUniqSupply, splitUniqSupply, @@ -156,12 +156,19 @@ mapUs f (x:xs) returnUs (r:rs) mapAndUnzipUs :: (a -> UniqSM (b,c)) -> [a] -> UniqSM ([b],[c]) +mapAndUnzip3Us :: (a -> UniqSM (b,c,d)) -> [a] -> UniqSM ([b],[c],[d]) mapAndUnzipUs f [] = returnUs ([],[]) mapAndUnzipUs f (x:xs) = f x `thenUs` \ (r1, r2) -> mapAndUnzipUs f xs `thenUs` \ (rs1, rs2) -> returnUs (r1:rs1, r2:rs2) + +mapAndUnzip3Us f [] = returnUs ([],[],[]) +mapAndUnzip3Us f (x:xs) + = f x `thenUs` \ (r1, r2, r3) -> + mapAndUnzip3Us f xs `thenUs` \ (rs1, rs2, rs3) -> + returnUs (r1:rs1, r2:rs2, r3:rs3) \end{code} %************************************************************************ diff --git a/ghc/compiler/coreSyn/AnnCoreSyn.lhs b/ghc/compiler/coreSyn/AnnCoreSyn.lhs index af16b22c52..9f51e1a4b1 100644 --- a/ghc/compiler/coreSyn/AnnCoreSyn.lhs +++ b/ghc/compiler/coreSyn/AnnCoreSyn.lhs @@ -1,5 +1,5 @@ % -% (c) The GRASP/AQUA Project, Glasgow University, 1992-1994 +% (c) The GRASP/AQUA Project, Glasgow University, 1992-1996 % \section[AnnCoreSyntax]{Annotated core syntax} @@ -16,106 +16,91 @@ module AnnCoreSyn ( AnnCoreCaseAlts(..), AnnCoreCaseDefault(..), deAnnotate -- we may eventually export some of the other deAnners - - -- and to make the interface self-sufficient ) where -import PrelInfo ( PrimOp(..), PrimRep - IF_ATTACK_PRAGMAS(COMMA tagOf_PrimOp) - IF_ATTACK_PRAGMAS(COMMA pprPrimOp) - ) -import Literal ( Literal ) +import Ubiq{-uitous-} + import CoreSyn -import Outputable -import CostCentre ( CostCentre ) -#if USE_ATTACK_PRAGMAS -import Util -#endif \end{code} \begin{code} -data AnnCoreBinding binder bindee annot - = AnnCoNonRec binder (AnnCoreExpr binder bindee annot) - | AnnCoRec [(binder, AnnCoreExpr binder bindee annot)] +data AnnCoreBinding val_bdr val_occ tyvar uvar annot + = AnnNonRec val_bdr (AnnCoreExpr val_bdr val_occ tyvar uvar annot) + | AnnRec [(val_bdr, AnnCoreExpr val_bdr val_occ tyvar uvar annot)] \end{code} \begin{code} -type AnnCoreExpr binder bindee annot = (annot, AnnCoreExpr' binder bindee annot) - -data AnnCoreExpr' binder bindee annot - = AnnCoVar bindee - | AnnCoLit Literal +type AnnCoreExpr val_bdr val_occ tyvar uvar annot + = (annot, AnnCoreExpr' val_bdr val_occ tyvar uvar annot) - | AnnCoCon Id [Type] [GenCoreAtom bindee] +data AnnCoreExpr' val_bdr val_occ tyvar uvar annot + = AnnVar val_occ + | AnnLit Literal - | AnnCoPrim PrimOp [Type] [GenCoreAtom bindee] + | AnnCon Id [GenCoreArg val_occ tyvar uvar] + | AnnPrim PrimOp [GenCoreArg val_occ tyvar uvar] - | AnnCoLam binder - (AnnCoreExpr binder bindee annot) - | AnnCoTyLam TyVar - (AnnCoreExpr binder bindee annot) + | AnnLam (GenCoreBinder val_bdr tyvar uvar) + (AnnCoreExpr val_bdr val_occ tyvar uvar annot) - | AnnCoApp (AnnCoreExpr binder bindee annot) - (GenCoreAtom bindee) - | AnnCoTyApp (AnnCoreExpr binder bindee annot) - Type + | AnnApp (AnnCoreExpr val_bdr val_occ tyvar uvar annot) + (GenCoreArg val_occ tyvar uvar) - | AnnCoCase (AnnCoreExpr binder bindee annot) - (AnnCoreCaseAlts binder bindee annot) + | AnnCase (AnnCoreExpr val_bdr val_occ tyvar uvar annot) + (AnnCoreCaseAlts val_bdr val_occ tyvar uvar annot) - | AnnCoLet (AnnCoreBinding binder bindee annot) - (AnnCoreExpr binder bindee annot) + | AnnLet (AnnCoreBinding val_bdr val_occ tyvar uvar annot) + (AnnCoreExpr val_bdr val_occ tyvar uvar annot) - | AnnCoSCC CostCentre - (AnnCoreExpr binder bindee annot) + | AnnSCC CostCentre + (AnnCoreExpr val_bdr val_occ tyvar uvar annot) \end{code} \begin{code} -data AnnCoreCaseAlts binder bindee annot - = AnnCoAlgAlts [(Id, - [binder], - AnnCoreExpr binder bindee annot)] - (AnnCoreCaseDefault binder bindee annot) - | AnnCoPrimAlts [(Literal, - AnnCoreExpr binder bindee annot)] - (AnnCoreCaseDefault binder bindee annot) - -data AnnCoreCaseDefault binder bindee annot - = AnnCoNoDefault - | AnnCoBindDefault binder - (AnnCoreExpr binder bindee annot) +data AnnCoreCaseAlts val_bdr val_occ tyvar uvar annot + = AnnAlgAlts [(Id, + [val_bdr], + AnnCoreExpr val_bdr val_occ tyvar uvar annot)] + (AnnCoreCaseDefault val_bdr val_occ tyvar uvar annot) + | AnnPrimAlts [(Literal, + AnnCoreExpr val_bdr val_occ tyvar uvar annot)] + (AnnCoreCaseDefault val_bdr val_occ tyvar uvar annot) + +data AnnCoreCaseDefault val_bdr val_occ tyvar uvar annot + = AnnNoDefault + | AnnBindDefault val_bdr + (AnnCoreExpr val_bdr val_occ tyvar uvar annot) \end{code} \begin{code} -deAnnotate :: AnnCoreExpr bndr bdee ann -> GenCoreExpr bndr bdee - -deAnnotate (_, AnnCoVar v) = Var v -deAnnotate (_, AnnCoLit lit) = Lit lit -deAnnotate (_, AnnCoCon con tys args) = Con con tys args -deAnnotate (_, AnnCoPrim op tys args) = Prim op tys args -deAnnotate (_, AnnCoLam binder body) = Lam binder (deAnnotate body) -deAnnotate (_, AnnCoTyLam tyvar body) = CoTyLam tyvar (deAnnotate body) -deAnnotate (_, AnnCoApp fun arg) = App (deAnnotate fun) arg -deAnnotate (_, AnnCoTyApp fun ty) = CoTyApp (deAnnotate fun) ty -deAnnotate (_, AnnCoSCC lbl body) = SCC lbl (deAnnotate body) - -deAnnotate (_, AnnCoLet bind body) +deAnnotate :: AnnCoreExpr val_bdr val_occ tyvar uvar ann + -> GenCoreExpr val_bdr val_occ tyvar uvar + +deAnnotate (_, AnnVar v) = Var v +deAnnotate (_, AnnLit lit) = Lit lit +deAnnotate (_, AnnCon con args) = Con con args +deAnnotate (_, AnnPrim op args) = Prim op args +deAnnotate (_, AnnLam binder body)= Lam binder (deAnnotate body) +deAnnotate (_, AnnApp fun arg) = App (deAnnotate fun) arg +deAnnotate (_, AnnSCC lbl body) = SCC lbl (deAnnotate body) + +deAnnotate (_, AnnLet bind body) = Let (deAnnBind bind) (deAnnotate body) where - deAnnBind (AnnCoNonRec var rhs) = NonRec var (deAnnotate rhs) - deAnnBind (AnnCoRec pairs) = Rec [(v,deAnnotate rhs) | (v,rhs) <- pairs] + deAnnBind (AnnNonRec var rhs) = NonRec var (deAnnotate rhs) + deAnnBind (AnnRec pairs) = Rec [(v,deAnnotate rhs) | (v,rhs) <- pairs] -deAnnotate (_, AnnCoCase scrut alts) +deAnnotate (_, AnnCase scrut alts) = Case (deAnnotate scrut) (deAnnAlts alts) where - deAnnAlts (AnnCoAlgAlts alts deflt) + deAnnAlts (AnnAlgAlts alts deflt) = AlgAlts [(con,args,deAnnotate rhs) | (con,args,rhs) <- alts] (deAnnDeflt deflt) - deAnnAlts (AnnCoPrimAlts alts deflt) + deAnnAlts (AnnPrimAlts alts deflt) = PrimAlts [(lit,deAnnotate rhs) | (lit,rhs) <- alts] (deAnnDeflt deflt) - deAnnDeflt AnnCoNoDefault = NoDefault - deAnnDeflt (AnnCoBindDefault var rhs) = BindDefault var (deAnnotate rhs) + deAnnDeflt AnnNoDefault = NoDefault + deAnnDeflt (AnnBindDefault var rhs) = BindDefault var (deAnnotate rhs) \end{code} diff --git a/ghc/compiler/coreSyn/CoreSyn.lhs b/ghc/compiler/coreSyn/CoreSyn.lhs index 1599273d24..037afb41f9 100644 --- a/ghc/compiler/coreSyn/CoreSyn.lhs +++ b/ghc/compiler/coreSyn/CoreSyn.lhs @@ -8,7 +8,7 @@ module CoreSyn ( GenCoreBinding(..), GenCoreExpr(..), - GenCoreArg(..),GenCoreBinder(..), GenCoreCaseAlts(..), + GenCoreArg(..), GenCoreBinder(..), GenCoreCaseAlts(..), GenCoreCaseDefault(..), bindersOf, pairsFromCoreBinds, rhssOfBind, @@ -17,9 +17,9 @@ module CoreSyn ( mkApp, mkCon, mkPrim, mkValLam, mkTyLam, mkUseLam, mkLam, - digForLambdas, + collectBinders, - collectArgs, isValArg, + collectArgs, isValArg, notValArg, numValArgs, mkCoLetAny, mkCoLetNoUnboxed, mkCoLetUnboxedToCase, mkCoLetsAny, mkCoLetsNoUnboxed, mkCoLetsUnboxedToCase, @@ -143,10 +143,10 @@ desugarer sets up constructors as applications of global @Vars@s. Ye olde abstraction and application operators. \begin{code} | Lam (GenCoreBinder val_bdr tyvar uvar) - (GenCoreExpr val_bdr val_occ tyvar uvar) + (GenCoreExpr val_bdr val_occ tyvar uvar) | App (GenCoreExpr val_bdr val_occ tyvar uvar) - (GenCoreArg val_occ tyvar uvar) + (GenCoreArg val_occ tyvar uvar) \end{code} Case expressions (\tr{case <expr> of <List of alternatives>}): there @@ -369,23 +369,23 @@ mkLam tyvars valvars body \end{code} We often want to strip off leading lambdas before getting down to -business. @digForLambdas@ is your friend. +business. @collectBinders@ is your friend. We expect (by convention) usage-, type-, and value- lambdas in that order. \begin{code} -digForLambdas :: +collectBinders :: GenCoreExpr val_bdr val_occ tyvar uvar -> ([uvar], [tyvar], [val_bdr], GenCoreExpr val_bdr val_occ tyvar uvar) -digForLambdas (Lam (UsageBinder u) body) +collectBinders (Lam (UsageBinder u) body) = let - (uvars, tyvars, args, final_body) = digForLambdas body + (uvars, tyvars, args, final_body) = collectBinders body in (u:uvars, tyvars, args, final_body) -digForLambdas other +collectBinders other = let (tyvars, args, body) = dig_for_tyvars other in @@ -468,6 +468,10 @@ is_Lit_or_Var a isValArg (LitArg _) = True -- often used for sanity-checking isValArg (VarArg _) = True isValArg _ = False + +notValArg = not . isValArg -- exists only because it's a common use of isValArg + +numValArgs as = length [ a | a <- as, isValArg a ] -- again, convenience \end{code} \begin{code} diff --git a/ghc/compiler/coreSyn/CoreUnfold.lhs b/ghc/compiler/coreSyn/CoreUnfold.lhs index 908c832705..7aec06e516 100644 --- a/ghc/compiler/coreSyn/CoreUnfold.lhs +++ b/ghc/compiler/coreSyn/CoreUnfold.lhs @@ -248,7 +248,7 @@ calcUnfoldingGuidance calcUnfoldingGuidance scc_s_OK bOMB_OUT_SIZE expr = let - (use_binders, ty_binders, val_binders, body) = digForLambdas expr + (use_binders, ty_binders, val_binders, body) = collectBinders expr in case (sizeExpr scc_s_OK bOMB_OUT_SIZE val_binders body) of @@ -292,7 +292,7 @@ sizeExpr scc_s_OK bOMB_OUT_SIZE args expr = if scc_s_OK then size_up body else Nothing size_up (Con con args) = -- 1 + # of val args - sizeN (1 + length [ va | va <- args, isValArg va ]) + sizeN (1 + numValArgs args) size_up (Prim op args) = sizeN op_cost -- NB: no charge for PrimOp args where op_cost = if primOpCanTriggerGC op @@ -303,7 +303,7 @@ sizeExpr scc_s_OK bOMB_OUT_SIZE args expr size_up expr@(Lam _ _) = let - (uvars, tyvars, args, body) = digForLambdas expr + (uvars, tyvars, args, body) = collectBinders expr in size_up body `addSizeN` length args @@ -528,7 +528,7 @@ ment_expr (Lit l) = consider_lit l ment_expr expr@(Lam _ _) = let - (uvars, tyvars, args, body) = digForLambdas expr + (uvars, tyvars, args, body) = collectBinders expr in extractIdsUf args `thenUf` \ bs_ids -> addInScopesUf bs_ids ( diff --git a/ghc/compiler/coreSyn/CoreUtils.lhs b/ghc/compiler/coreSyn/CoreUtils.lhs index 1a993e6a7e..363cecb61f 100644 --- a/ghc/compiler/coreSyn/CoreUtils.lhs +++ b/ghc/compiler/coreSyn/CoreUtils.lhs @@ -15,8 +15,8 @@ module CoreUtils ( , mkErrorApp, escErrorMsg , argToExpr , unTagBinders, unTagBindersAlts + , manifestlyWHNF, manifestlyBottom {- exprSmallEnoughToDup, - manifestlyWHNF, manifestlyBottom, coreExprArity, isWrapperFor, maybeErrorApp, @@ -31,11 +31,12 @@ import IdLoop -- for pananoia-checking purposes import CoreSyn import CostCentre ( isDictCC ) -import Id ( idType, mkSysLocal, +import Id ( idType, mkSysLocal, getIdArity, isBottomingId, addOneToIdEnv, growIdEnvList, lookupIdEnv, isNullIdEnv, IdEnv(..), GenId{-instances-} ) +import IdInfo ( arityMaybe ) import Literal ( literalType, isNoRepLit, Literal(..) ) import Maybes ( catMaybes ) import PprCore ( GenCoreExpr{-instances-}, GenCoreArg{-instances-} ) @@ -259,6 +260,7 @@ exprSmallEnoughToDup expr -- for now, just: <var> applied to <args> && length args <= 6 -- or 10 or 1 or 4 or anything smallish. _ -> False } +-} \end{code} Question (ADR): What is the above used for? Is a _ccall_ really small enough? @@ -269,29 +271,31 @@ errs on the conservative side (returning \tr{False})---I've probably left something out... [WDP] \begin{code} -manifestlyWHNF :: GenCoreExpr bndr Id -> Bool +manifestlyWHNF :: GenCoreExpr bndr Id tyvar uvar -> Bool + +manifestlyWHNF (Var _) = True +manifestlyWHNF (Lit _) = True +manifestlyWHNF (Con _ _) = True +manifestlyWHNF (SCC _ e) = manifestlyWHNF e +manifestlyWHNF (Let _ e) = False +manifestlyWHNF (Case _ _) = False -manifestlyWHNF (Var _) = True -manifestlyWHNF (Lit _) = True -manifestlyWHNF (Con _ _ _) = True -- ToDo: anything for Prim? -manifestlyWHNF (Lam _ _) = True -manifestlyWHNF (CoTyLam _ e) = manifestlyWHNF e -manifestlyWHNF (SCC _ e) = manifestlyWHNF e -manifestlyWHNF (Let _ e) = False -manifestlyWHNF (Case _ _) = False +manifestlyWHNF (Lam (ValBinder _) _) = True +manifestlyWHNF (Lam other_binder e) = manifestlyWHNF e manifestlyWHNF other_expr -- look for manifest partial application = case (collectArgs other_expr) of { (fun, args) -> case fun of - Var f -> let - num_val_args = length [ a | (ValArg a) <- args ] - in - num_val_args == 0 || -- Just a type application of - -- a variable (f t1 t2 t3) - -- counts as WHNF - case (arityMaybe (getIdArity f)) of - Nothing -> False - Just arity -> num_val_args < arity + Var f -> let + num_val_args = numValArgs args + in + num_val_args == 0 -- Just a type application of + -- a variable (f t1 t2 t3); + -- counts as WHNF. + || + case (arityMaybe (getIdArity f)) of + Nothing -> False + Just arity -> num_val_args < arity _ -> False } @@ -303,17 +307,19 @@ some point. It isn't a disaster if it errs on the conservative side (returning \tr{False}). \begin{code} -manifestlyBottom :: GenCoreExpr bndr Id -> Bool +manifestlyBottom :: GenCoreExpr bndr Id tyvar uvar -> Bool manifestlyBottom (Var v) = isBottomingId v manifestlyBottom (Lit _) = False -manifestlyBottom (Con _ _ _) = False -manifestlyBottom (Prim _ _ _)= False -manifestlyBottom (Lam _ _) = False -- we do not assume \x.bottom == bottom. should we? ToDo -manifestlyBottom (CoTyLam _ e) = manifestlyBottom e +manifestlyBottom (Con _ _) = False +manifestlyBottom (Prim _ _) = False manifestlyBottom (SCC _ e) = manifestlyBottom e manifestlyBottom (Let _ e) = manifestlyBottom e + -- We do not assume \x.bottom == bottom: +manifestlyBottom (Lam (ValBinder _) _) = False +manifestlyBottom (Lam other_binder e) = manifestlyBottom e + manifestlyBottom (Case e a) = manifestlyBottom e || (case a of @@ -331,15 +337,16 @@ manifestlyBottom (Case e a) manifestlyBottom other_expr -- look for manifest partial application = case (collectArgs other_expr) of { (fun, args) -> case fun of - Var f | isBottomingId f -> True -- Application of a function which - -- always gives bottom; we treat this as - -- a WHNF, because it certainly doesn't - -- need to be shared! + Var f | isBottomingId f -> True + -- Application of a function which always gives + -- bottom; we treat this as a WHNF, because it + -- certainly doesn't need to be shared! _ -> False } \end{code} \begin{code} +{-LATER: coreExprArity :: (Id -> Maybe (GenCoreExpr bndr Id)) -> GenCoreExpr bndr Id @@ -371,7 +378,7 @@ Probably a little too HACKY [WDP]. isWrapperFor :: CoreExpr -> Id -> Bool expr `isWrapperFor` var - = case (digForLambdas expr) of { (_, _, args, body) -> -- lambdas off the front + = case (collectBinders expr) of { (_, _, args, body) -> -- lambdas off the front unravel_casing args body --NO, THANKS: && not (null args) } diff --git a/ghc/compiler/coreSyn/FreeVars.lhs b/ghc/compiler/coreSyn/FreeVars.lhs index 62c8e80de2..8879ffeaf1 100644 --- a/ghc/compiler/coreSyn/FreeVars.lhs +++ b/ghc/compiler/coreSyn/FreeVars.lhs @@ -1,5 +1,5 @@ % -% (c) The GRASP/AQUA Project, Glasgow University, 1992-1995 +% (c) The GRASP/AQUA Project, Glasgow University, 1992-1996 % Taken quite directly from the Peyton Jones/Lester paper. @@ -18,24 +18,28 @@ module FreeVars ( CoreExprWithFVs(..), -- For the above functions AnnCoreExpr(..), -- Dito FVInfo(..), LeakInfo(..) - - -- and to make the interface self-sufficient... ) where +import Ubiq{-uitous-} import AnnCoreSyn -- output -import PrelInfo ( PrimOp(..), PrimRep -- for CCallOp - IF_ATTACK_PRAGMAS(COMMA tagOf_PrimOp) - IF_ATTACK_PRAGMAS(COMMA pprPrimOp) +import CoreSyn +import Id ( idType, getIdArity, isBottomingId, + emptyIdSet, singletonIdSet, mkIdSet, + elementOfIdSet, minusIdSet, unionManyIdSets, + IdSet(..) + ) +import IdInfo ( arityMaybe ) +import PrimOp ( PrimOp(..) ) +import Type ( tyVarsOfType ) +import TyVar ( emptyTyVarSet, singletonTyVarSet, minusTyVarSet, + intersectTyVarSets, + TyVarSet(..) ) -import Type ( extractTyVarsFromTy ) -import Id ( idType, getIdArity, toplevelishId, isBottomingId ) -import IdInfo -- Wanted for arityMaybe, but it seems you have - -- to import it all... (Death to the Instance Virus!) -import Maybes -import UniqSet -import Util +import UniqSet ( unionUniqSets ) +import Usage ( UVar(..) ) +import Util ( panic, assertPanic ) \end{code} %************************************************************************ @@ -55,35 +59,36 @@ I've half-convinced myself we don't for case- and letrec bound ids but I might be wrong. (SLPJ, date unknown) \begin{code} -type CoreExprWithFVs = AnnCoreExpr Id Id FVInfo +type CoreExprWithFVs = AnnCoreExpr Id Id TyVar UVar FVInfo type TyVarCands = TyVarSet -- for when we carry around lists of type IdCands = IdSet -- "candidate" TyVars/Ids. -noTyVarCands = emptyUniqSet -noIdCands = emptyUniqSet - -data FVInfo = FVInfo - IdSet -- Free ids - TyVarSet -- Free tyvars - LeakInfo - -noFreeIds = emptyUniqSet -noFreeTyVars = emptyUniqSet -aFreeId i = singletonUniqSet i -aFreeTyVar t = singletonUniqSet t -is_among = elementOfUniqSet -combine = unionUniqSets -munge_id_ty i = mkUniqSet (extractTyVarsFromTy (idType i)) +noTyVarCands = emptyTyVarSet +noIdCands = emptyIdSet + +data FVInfo + = FVInfo IdSet -- Free ids + TyVarSet -- Free tyvars + LeakInfo + +noFreeIds = emptyIdSet +noFreeTyVars = emptyTyVarSet +noFreeAnything = (noFreeIds, noFreeTyVars) +aFreeId i = singletonIdSet i +aFreeTyVar t = singletonTyVarSet t +is_among = elementOfIdSet +munge_id_ty i = tyVarsOfType (idType i) +combine = unionUniqSets -- used both for {Id,TyVar}Sets combineFVInfo (FVInfo fvs1 tfvs1 leak1) (FVInfo fvs2 tfvs2 leak2) = FVInfo (fvs1 `combine` fvs2) (tfvs1 `combine` tfvs2) - (leak1 `orLeak` leak2) + (leak1 `orLeak` leak2) \end{code} -Leak-free-ness is based only on the value, not the type. -In particular, nested collections of constructors are guaranteed leak free. -Function applications are not, except for PAPs. +Leak-free-ness is based only on the value, not the type. In +particular, nested collections of constructors are guaranteed leak +free. Function applications are not, except for PAPs. Applications of error gets (LeakFree bigArity) -- a hack! @@ -111,7 +116,11 @@ freeVars :: CoreExpr -> CoreExprWithFVs freeVars expr = fvExpr noIdCands noTyVarCands expr \end{code} +%************************************************************************ +%* * \subsection{Free variables (and types)} +%* * +%************************************************************************ We do the free-variable stuff by passing around ``candidates lists'' of @Ids@ and @TyVars@ that may be considered free. This is useful, @@ -131,7 +140,7 @@ fvExpr id_cands tyvar_cands (Var v) else noFreeIds) noFreeTyVars leakiness, - AnnCoVar v) + AnnVar v) where leakiness | isBottomingId v = lEAK_FREE_BIG -- Hack @@ -140,96 +149,94 @@ fvExpr id_cands tyvar_cands (Var v) Just arity -> LeakFree arity fvExpr id_cands tyvar_cands (Lit k) - = (FVInfo noFreeIds noFreeTyVars lEAK_FREE_0, AnnCoLit k) + = (FVInfo noFreeIds noFreeTyVars lEAK_FREE_0, AnnLit k) -fvExpr id_cands tyvar_cands (Con c tys args) - = (FVInfo args_fvs tfvs lEAK_FREE_0, AnnCoCon c tys args) +fvExpr id_cands tyvar_cands (Con c args) + = (FVInfo args_fvs tfvs lEAK_FREE_0, AnnCon c args) where - args_fvs = foldr (combine . freeAtom id_cands) noFreeIds args - tfvs = foldr (combine . freeTy tyvar_cands) noFreeTyVars tys + (args_fvs, tfvs) = freeArgs id_cands tyvar_cands args -fvExpr id_cands tyvar_cands (Prim op@(CCallOp _ _ _ _ res_ty) tys args) - = ASSERT (null tys) - (FVInfo args_fvs tfvs lEAK_FREE_0, AnnCoPrim op tys args) +fvExpr id_cands tyvar_cands (Prim op args) + = (FVInfo args_fvs tfvs lEAK_FREE_0, AnnPrim op args) where - args_fvs = foldr (combine . freeAtom id_cands) noFreeIds args - tfvs = foldr (combine . freeTy tyvar_cands) noFreeTyVars (res_ty:tys) + (args_fvs, tfvs) = freeArgs id_cands tyvar_cands args_to_use{-NB-} + args_to_use + = case op of + CCallOp _ _ _ _ res_ty -> TyArg res_ty : args + _ -> args -fvExpr id_cands tyvar_cands (Prim op tys args) - = (FVInfo args_fvs tfvs lEAK_FREE_0, AnnCoPrim op tys args) - where - args_fvs = foldr (combine . freeAtom id_cands) noFreeIds args - tfvs = foldr (combine . freeTy tyvar_cands) noFreeTyVars tys +-- this Lam stuff could probably be improved by rewriting (WDP 96/03) + +fvExpr id_cands tyvar_cands (Lam (UsageBinder uvar) body) + = panic "fvExpr:Lam UsageBinder" -fvExpr id_cands tyvar_cands (Lam binder body) - = (FVInfo (freeVarsOf body2 `minusUniqSet` singletonUniqSet binder) - (freeTyVarsOf body2 `combine` munge_id_ty binder) +fvExpr id_cands tyvar_cands (Lam b@(ValBinder binder) body) + = (FVInfo (freeVarsOf body2 `minusIdSet` singletonIdSet binder) + (freeTyVarsOf body2 `combine` munge_id_ty binder) leakiness, - AnnCoLam binder body2) + AnnLam b body2) where -- We need to collect free tyvars from the binders - body2 = fvExpr (singletonUniqSet binder `combine` id_cands) tyvar_cands body + body2 = fvExpr (singletonIdSet binder `combine` id_cands) tyvar_cands body leakiness = case leakinessOf body2 of MightLeak -> LeakFree 1 LeakFree n -> LeakFree (n + 1) -fvExpr id_cands tyvar_cands (CoTyLam tyvar body) +fvExpr id_cands tyvar_cands (Lam b@(TyBinder tyvar) body) = (FVInfo (freeVarsOf body2) - (freeTyVarsOf body2 `minusUniqSet` aFreeTyVar tyvar) + (freeTyVarsOf body2 `minusTyVarSet` aFreeTyVar tyvar) (leakinessOf body2), - AnnCoTyLam tyvar body2) + AnnLam b body2) where body2 = fvExpr id_cands (aFreeTyVar tyvar `combine` tyvar_cands) body +-- ditto on rewriting this App stuff (WDP 96/03) + fvExpr id_cands tyvar_cands (App fun arg) - = (FVInfo (freeVarsOf fun2 `combine` fvs_arg) - (freeTyVarsOf fun2) + = (FVInfo (freeVarsOf fun2 `combine` fvs_arg) + (freeTyVarsOf fun2 `combine` tfvs_arg) leakiness, - AnnCoApp fun2 arg) + AnnApp fun2 arg) where fun2 = fvExpr id_cands tyvar_cands fun - fvs_arg = freeAtom id_cands arg + fun2_leakiness = leakinessOf fun2 - leakiness = case leakinessOf fun2 of - LeakFree n | n>1 -> LeakFree (n-1) -- Note > not >= - other -> MightLeak + (fvs_arg, tfvs_arg) = freeArgs id_cands tyvar_cands [arg] -fvExpr id_cands tyvar_cands (CoTyApp expr ty) - = (FVInfo (freeVarsOf expr2) - (freeTyVarsOf expr2 `combine` tfvs_arg) - (leakinessOf expr2), - AnnCoTyApp expr2 ty) - where - expr2 = fvExpr id_cands tyvar_cands expr - tfvs_arg = freeTy tyvar_cands ty + leakiness = if (notValArg arg) then + fun2_leakiness + else + case fun2_leakiness of + LeakFree n | n>1 -> LeakFree (n-1) -- Note > not >= + other -> MightLeak fvExpr id_cands tyvar_cands (Case expr alts) = (combineFVInfo expr_fvinfo alts_fvinfo, - AnnCoCase expr2 alts') + AnnCase expr2 alts') where expr2@(expr_fvinfo,_) = fvExpr id_cands tyvar_cands expr (alts_fvinfo, alts') = annotate_alts alts annotate_alts (AlgAlts alts deflt) - = (fvinfo, AnnCoAlgAlts alts' deflt') + = (fvinfo, AnnAlgAlts alts' deflt') where (alts_fvinfo_s, alts') = unzip (map ann_boxed_alt alts) (deflt_fvinfo, deflt') = annotate_default deflt fvinfo = foldr combineFVInfo deflt_fvinfo alts_fvinfo_s ann_boxed_alt (con, params, rhs) - = (FVInfo (freeVarsOf rhs' `minusUniqSet` mkUniqSet params) + = (FVInfo (freeVarsOf rhs' `minusIdSet` mkIdSet params) (freeTyVarsOf rhs' `combine` param_ftvs) (leakinessOf rhs'), (con, params, rhs')) where - rhs' = fvExpr (mkUniqSet params `combine` id_cands) tyvar_cands rhs + rhs' = fvExpr (mkIdSet params `combine` id_cands) tyvar_cands rhs param_ftvs = foldr (combine . munge_id_ty) noFreeTyVars params -- We need to collect free tyvars from the binders annotate_alts (PrimAlts alts deflt) - = (fvinfo, AnnCoPrimAlts alts' deflt') + = (fvinfo, AnnPrimAlts alts' deflt') where (alts_fvinfo_s, alts') = unzip (map ann_unboxed_alt alts) (deflt_fvinfo, deflt') = annotate_default deflt @@ -240,13 +247,13 @@ fvExpr id_cands tyvar_cands (Case expr alts) rhs'@(rhs_info,_) = fvExpr id_cands tyvar_cands rhs annotate_default NoDefault = (FVInfo noFreeIds noFreeTyVars lEAK_FREE_BIG, - AnnCoNoDefault) + AnnNoDefault) annotate_default (BindDefault binder rhs) - = (FVInfo (freeVarsOf rhs' `minusUniqSet` aFreeId binder) + = (FVInfo (freeVarsOf rhs' `minusIdSet` aFreeId binder) (freeTyVarsOf rhs' `combine` binder_ftvs) (leakinessOf rhs'), - AnnCoBindDefault binder rhs') + AnnBindDefault binder rhs') where rhs' = fvExpr (aFreeId binder `combine` id_cands) tyvar_cands rhs binder_ftvs = munge_id_ty binder @@ -256,11 +263,11 @@ fvExpr id_cands tyvar_cands (Let (NonRec binder rhs) body) = (FVInfo (freeVarsOf rhs' `combine` body_fvs) (freeTyVarsOf rhs' `combine` freeTyVarsOf body2 `combine` binder_ftvs) (leakinessOf rhs' `orLeak` leakinessOf body2), - AnnCoLet (AnnCoNonRec binder rhs') body2) + AnnLet (AnnNonRec binder rhs') body2) where rhs' = fvExpr id_cands tyvar_cands rhs body2 = fvExpr (aFreeId binder `combine` id_cands) tyvar_cands body - body_fvs = freeVarsOf body2 `minusUniqSet` aFreeId binder + body_fvs = freeVarsOf body2 `minusIdSet` aFreeId binder binder_ftvs = munge_id_ty binder -- We need to collect free tyvars from the binder @@ -268,38 +275,56 @@ fvExpr id_cands tyvar_cands (Let (Rec binds) body) = (FVInfo (binds_fvs `combine` body_fvs) (rhss_tfvs `combine` freeTyVarsOf body2 `combine` binders_ftvs) (leakiness_of_rhss `orLeak` leakinessOf body2), - AnnCoLet (AnnCoRec (binders `zip` rhss')) body2) + AnnLet (AnnRec (binders `zip` rhss')) body2) where (binders, rhss) = unzip binds new_id_cands = binders_set `combine` id_cands - binders_set = mkUniqSet binders + binders_set = mkIdSet binders rhss' = map (fvExpr new_id_cands tyvar_cands) rhss FVInfo rhss_fvs rhss_tfvs leakiness_of_rhss = foldr1 combineFVInfo [info | (info,_) <- rhss'] - binds_fvs = rhss_fvs `minusUniqSet` binders_set + binds_fvs = rhss_fvs `minusIdSet` binders_set body2 = fvExpr new_id_cands tyvar_cands body - body_fvs = freeVarsOf body2 `minusUniqSet` binders_set + body_fvs = freeVarsOf body2 `minusIdSet` binders_set binders_ftvs = foldr (combine . munge_id_ty) noFreeTyVars binders -- We need to collect free tyvars from the binders fvExpr id_cands tyvar_cands (SCC label expr) - = (fvinfo, AnnCoSCC label expr2) + = (fvinfo, AnnSCC label expr2) where expr2@(fvinfo,_) = fvExpr id_cands tyvar_cands expr \end{code} \begin{code} -freeAtom :: IdCands -> CoreArg -> IdSet - -freeAtom cands (LitArg k) = noFreeIds -freeAtom cands (VarArg v) | v `is_among` cands = aFreeId v - | otherwise = noFreeIds +freeArgs :: IdCands -> TyVarCands + -> [CoreArg] + -> (IdSet, TyVarSet) +freeArgs icands tcands [] = noFreeAnything +freeArgs icands tcands (arg:args) + -- this code is written this funny way only for "efficiency" purposes + = let + free_first_arg@(arg_fvs, tfvs) = free_arg arg + in + if (null args) then + free_first_arg + else + case (freeArgs icands tcands args) of { (irest, trest) -> + (arg_fvs `combine` irest, tfvs `combine` trest) } + where + free_arg (LitArg _) = noFreeAnything + free_arg (UsageArg _) = noFreeAnything + free_arg (TyArg ty) = (noFreeIds, freeTy tcands ty) + free_arg (VarArg v) + | v `is_among` icands = (aFreeId v, noFreeTyVars) + | otherwise = noFreeAnything + +--------- freeTy :: TyVarCands -> Type -> TyVarSet -freeTy cands ty = mkUniqSet (extractTyVarsFromTy ty) `intersectUniqSets` cands +freeTy cands ty = tyVarsOfType ty `intersectTyVarSets` cands freeVarsOf :: CoreExprWithFVs -> IdSet freeVarsOf (FVInfo free_vars _ _, _) = free_vars @@ -348,8 +373,8 @@ As it happens this is only ever used by the Specialiser! \begin{code} type FVCoreBinder = (Id, IdSet) -type FVCoreExpr = GenCoreExpr FVCoreBinder Id -type FVCoreBinding = GenCoreBinding FVCoreBinder Id +type FVCoreExpr = GenCoreExpr FVCoreBinder Id TyVar UVar +type FVCoreBinding = GenCoreBinding FVCoreBinder Id TyVar UVar type InterestingIdFun = IdSet -- Non-top-level in-scope variables @@ -370,38 +395,31 @@ addExprFVs fv_cand in_scope (Var v) addExprFVs fv_cand in_scope (Lit lit) = (Lit lit, noFreeIds) -addExprFVs fv_cand in_scope (Con con tys args) - = (Con con tys args, +addExprFVs fv_cand in_scope (Con con args) + = (Con con args, if fv_cand in_scope con then aFreeId con - else noFreeIds - `combine` - unionManyUniqSets (map (fvsOfAtom fv_cand in_scope) args)) + else noFreeIds `combine` fvsOfArgs fv_cand in_scope args) -addExprFVs fv_cand in_scope (Prim op tys args) - = (Prim op tys args, - unionManyUniqSets (map (fvsOfAtom fv_cand in_scope) args)) +addExprFVs fv_cand in_scope (Prim op args) + = (Prim op args, fvsOfArgs fv_cand in_scope args) addExprFVs fv_cand in_scope (Lam binder body) - = (Lam (binder,lam_fvs) new_body, lam_fvs) + = (Lam new_binder new_body, lam_fvs) where - binder_set = singletonUniqSet binder - new_in_scope = in_scope `combine` binder_set + (new_binder, binder_set) + = case binder of + TyBinder t -> (TyBinder t, emptyIdSet) + UsageBinder u -> (UsageBinder u, emptyIdSet) + ValBinder b -> (ValBinder (b, lam_fvs), + singletonIdSet b) + + new_in_scope = in_scope `combine` binder_set (new_body, body_fvs) = addExprFVs fv_cand new_in_scope body - lam_fvs = body_fvs `minusUniqSet` binder_set - -addExprFVs fv_cand in_scope (CoTyLam tyvar body) - = (CoTyLam tyvar body2, body_fvs) - where - (body2, body_fvs) = addExprFVs fv_cand in_scope body + lam_fvs = body_fvs `minusIdSet` binder_set addExprFVs fv_cand in_scope (App fun arg) - = (App fun2 arg, fun_fvs `combine` fvsOfAtom fv_cand in_scope arg) - where - (fun2, fun_fvs) = addExprFVs fv_cand in_scope fun - -addExprFVs fv_cand in_scope (CoTyApp fun ty) - = (CoTyApp fun2 ty, fun_fvs) + = (App fun2 arg, fun_fvs `combine` fvsOfArgs fv_cand in_scope [arg]) where (fun2, fun_fvs) = addExprFVs fv_cand in_scope fun @@ -416,13 +434,13 @@ addExprFVs fv_cand in_scope (Case scrut alts) where (alg_alts', alt_fvs) = unzip (map do_alg_alt alg_alts) (deflt', deflt_fvs) = do_deflt deflt - fvs = unionManyUniqSets (deflt_fvs : alt_fvs) + fvs = unionManyIdSets (deflt_fvs : alt_fvs) PrimAlts prim_alts deflt -> (PrimAlts prim_alts' deflt', fvs) where (prim_alts', alt_fvs) = unzip (map do_prim_alt prim_alts) (deflt', deflt_fvs) = do_deflt deflt - fvs = unionManyUniqSets (deflt_fvs : alt_fvs) + fvs = unionManyIdSets (deflt_fvs : alt_fvs) do_alg_alt :: (Id, [Id], CoreExpr) -> ((Id, [FVCoreBinder], FVCoreExpr), IdSet) @@ -431,8 +449,8 @@ addExprFVs fv_cand in_scope (Case scrut alts) where new_in_scope = in_scope `combine` arg_set (rhs', rhs_fvs) = addExprFVs fv_cand new_in_scope rhs - fvs = rhs_fvs `minusUniqSet` arg_set - arg_set = mkUniqSet args + fvs = rhs_fvs `minusIdSet` arg_set + arg_set = mkIdSet args do_prim_alt (lit, rhs) = ((lit, rhs'), rhs_fvs) where @@ -444,11 +462,11 @@ addExprFVs fv_cand in_scope (Case scrut alts) where new_in_scope = in_scope `combine` var_set (rhs', rhs_fvs) = addExprFVs fv_cand new_in_scope rhs - fvs = rhs_fvs `minusUniqSet` var_set + fvs = rhs_fvs `minusIdSet` var_set var_set = aFreeId var addExprFVs fv_cand in_scope (Let binds body) - = (Let binds' body2, fvs_binds `combine` (fvs_body `minusUniqSet` binder_set)) + = (Let binds' body2, fvs_binds `combine` (fvs_body `minusIdSet` binder_set)) where (binds', fvs_binds, new_in_scope, binder_set) = addBindingFVs fv_cand in_scope binds @@ -479,10 +497,10 @@ addBindingFVs fv_cand in_scope (NonRec binder rhs) binder_set = aFreeId binder addBindingFVs fv_cand in_scope (Rec pairs) - = (Rec pairs', unionManyUniqSets fvs_s, new_in_scope, binder_set) + = (Rec pairs', unionManyIdSets fvs_s, new_in_scope, binder_set) where binders = [binder | (binder,_) <- pairs] - binder_set = mkUniqSet binders + binder_set = mkIdSet binders new_in_scope = in_scope `combine` binder_set (pairs', fvs_s) = unzip (map (do_pair fv_cand new_in_scope binder_set) pairs) \end{code} @@ -504,17 +522,22 @@ addTopBindsFVs fv_cand (b:bs) \end{code} \begin{code} -fvsOfAtom :: InterestingIdFun -- "Interesting id" predicate +fvsOfArgs :: InterestingIdFun -- "Interesting id" predicate -> IdSet -- In scope ids - -> CoreArg + -> [CoreArg] -> IdSet -fvsOfAtom fv_cand in_scope (VarArg v) - = if fv_cand in_scope v - then aFreeId v - else noFreeIds -fvsOfAtom _ _ _ = noFreeIds -- if a literal... +fvsOfArgs _ _ [] = noFreeIds + +fvsOfArgs fv_cand in_scope [VarArg v] -- this is only a short-cut... + = if (fv_cand in_scope v) then aFreeId v else noFreeIds +fvsOfArgs _ _ [ _ ] = noFreeIds + +fvsOfArgs fv_cand in_scope args + = mkIdSet [ v | (VarArg v) <- args, fv_cand in_scope v ] + -- all other types of args are uninteresting here... +---------- do_pair :: InterestingIdFun -- "Interesting id" predicate -> IdSet -- In scope ids -> IdSet @@ -525,5 +548,5 @@ do_pair fv_cand in_scope binder_set (binder,rhs) = (((binder, fvs), rhs'), fvs) where (rhs', rhs_fvs) = addExprFVs fv_cand in_scope rhs - fvs = rhs_fvs `minusUniqSet` binder_set + fvs = rhs_fvs `minusIdSet` binder_set \end{code} diff --git a/ghc/compiler/coreSyn/PprCore.lhs b/ghc/compiler/coreSyn/PprCore.lhs index b3569e8866..770e9bf0e1 100644 --- a/ghc/compiler/coreSyn/PprCore.lhs +++ b/ghc/compiler/coreSyn/PprCore.lhs @@ -335,7 +335,7 @@ ppr_expr pe (Prim prim args) ppr_expr pe expr@(Lam _ _) = let - (uvars, tyvars, vars, body) = digForLambdas expr + (uvars, tyvars, vars, body) = collectBinders expr in ppHang (ppCat [pp_vars SLIT("_/u\\_") (puvar pe) uvars, pp_vars SLIT("_/\\_") (ptyvar pe) tyvars, diff --git a/ghc/compiler/deSugar/DsBinds.lhs b/ghc/compiler/deSugar/DsBinds.lhs index 691e086058..bc26cf44ec 100644 --- a/ghc/compiler/deSugar/DsBinds.lhs +++ b/ghc/compiler/deSugar/DsBinds.lhs @@ -34,13 +34,13 @@ import ListSetOps ( minusList, intersectLists ) import PprType ( GenType, GenTyVar ) import PprStyle ( PprStyle(..) ) import Pretty ( ppShow ) -import Type ( mkTyVarTy, splitSigmaTy ) -import TyVar ( GenTyVar ) +import Type ( mkTyVarTys, splitSigmaTy, + tyVarsOfType, tyVarsOfTypes + ) +import TyVar ( tyVarSetToList, GenTyVar ) import Unique ( Unique ) import Util ( isIn, panic ) -extractTyVarsFromTy = panic "DsBinds.extractTyVarsFromTy" -extractTyVarsFromTys = panic "DsBinds.extractTyVarsFromTys" isDictTy = panic "DsBinds.isDictTy" quantifyTy = panic "DsBinds.quantifyTy" \end{code} @@ -158,7 +158,7 @@ dsBinds (AbsBinds tyvars [] local_global_prs inst_binds val_binds) binders = collectTypedBinders val_binds mk_poly_private_binder id = newSysLocalDs (snd (quantifyTy tyvars (idType id))) - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars \end{code} @@ -240,10 +240,10 @@ dsBinds (AbsBinds all_tyvars dicts local_global_prs dict_binds val_binds) returnDs [ NonRec binder rhs | (binder,rhs) <- core_bind_prs ] where locals = [local | (local,global) <- local_global_prs] - non_ov_tyvar_tys = map mkTyVarTy non_overloaded_tyvars + non_ov_tyvar_tys = mkTyVarTys non_overloaded_tyvars - overloaded_tyvars = extractTyVarsFromTys (map idType dicts) - non_overloaded_tyvars = all_tyvars `minusList` overloaded_tyvars + overloaded_tyvars = tyVarsOfTypes (map idType dicts) + non_overloaded_tyvars = all_tyvars `minusList` (tyVarSetToList{-????-} overloaded_tyvars) binders = collectTypedBinders val_binds mk_binder id = newSysLocalDs (snd (quantifyTy non_overloaded_tyvars (idType id))) @@ -266,7 +266,7 @@ mkSatTyApp id tys = returnDs ty_app -- Common case | otherwise = newTyVarsDs (drop (length tys) tvs) `thenDs` \ tyvars -> - returnDs (mkTyLam tyvars (mkTyApp ty_app (map mkTyVarTy tyvars))) + returnDs (mkTyLam tyvars (mkTyApp ty_app (mkTyVarTys tyvars))) where (tvs, theta, tau_ty) = splitSigmaTy (idType id) ty_app = mkTyApp (Var id) tys @@ -351,8 +351,8 @@ dsInstBinds tyvars ((inst, expr) : bs) subst_item : subst_env) where inst_ty = idType inst - abs_tyvars = extractTyVarsFromTy inst_ty `intersectLists` tyvars - abs_tys = map mkTyVarTy abs_tyvars + abs_tyvars = tyVarsOfType inst_ty `intersectLists` tyvars + abs_tys = mkTyVarTys abs_tyvars (_, poly_inst_ty) = quantifyTy abs_tyvars inst_ty ------------------------ diff --git a/ghc/compiler/deSugar/DsListComp.lhs b/ghc/compiler/deSugar/DsListComp.lhs index 39b00d4a2c..7b6651a14e 100644 --- a/ghc/compiler/deSugar/DsListComp.lhs +++ b/ghc/compiler/deSugar/DsListComp.lhs @@ -67,7 +67,7 @@ dsListComp expr quals new_alpha_tyvar :: DsM (TyVar, Type) new_alpha_tyvar = newTyVarsDs [alphaTyVar] `thenDs` \ [new_ty] -> - returnDs (new_ty,mkTyVarTy new_ty) + returnDs (new_ty, mkTyVarTy new_ty) \end{code} %************************************************************************ diff --git a/ghc/compiler/deSugar/DsUtils.lhs b/ghc/compiler/deSugar/DsUtils.lhs index b58c6d5ebc..07cbe0b249 100644 --- a/ghc/compiler/deSugar/DsUtils.lhs +++ b/ghc/compiler/deSugar/DsUtils.lhs @@ -42,7 +42,7 @@ import PrelInfo ( stringTy ) import Id ( idType, getInstantiatedDataConSig, mkTupleCon, DataCon(..), DictVar(..), Id(..), GenId ) import TyCon ( mkTupleTyCon ) -import Type ( mkTyVarTy, mkRhoTy, mkFunTys, +import Type ( mkTyVarTys, mkRhoTy, mkFunTys, applyTyCon, getAppDataTyCon ) import UniqSet ( mkUniqSet, minusUniqSet, uniqSetToList, UniqSet(..) ) import Util ( panic, assertPanic ) @@ -400,7 +400,7 @@ mkTupleBind tyvars dicts local_global_prs tuple_expr globals = [global | (local,global) <- local_global_prs] no_of_binders = length local_global_prs - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars tuple_var_ty :: Type tuple_var_ty diff --git a/ghc/compiler/deforest/Cyclic.lhs b/ghc/compiler/deforest/Cyclic.lhs index 62f1fe0470..08b65d78bf 100644 --- a/ghc/compiler/deforest/Cyclic.lhs +++ b/ghc/compiler/deforest/Cyclic.lhs @@ -13,7 +13,7 @@ > import DefUtils > import Def2Core ( d2c, defPanic ) -> import Type ( glueTyArgs, quantifyTy, mkForallTy, mkTyVarTy, +> import Type ( glueTyArgs, quantifyTy, mkForallTy, mkTyVarTys, > TyVarTemplate > ) > import Digraph ( dfs ) @@ -372,7 +372,7 @@ expressions and function right hand sides that call this function. > mkLoopFunApp val_args ty_args f = > foldl App > (foldl CoTyApp (Var (DefArgVar f)) -> (map mkTyVarTy ty_args)) +> (mkTyVarTys ty_args)) > (map mkVar val_args) ----------------------------------------------------------------------------- diff --git a/ghc/compiler/deforest/DefUtils.lhs b/ghc/compiler/deforest/DefUtils.lhs index 54f8eeb118..2170ecaceb 100644 --- a/ghc/compiler/deforest/DefUtils.lhs +++ b/ghc/compiler/deforest/DefUtils.lhs @@ -21,7 +21,7 @@ >#endif > import Type ( cloneTyVar, mkTyVarTy, applyTypeEnvToTy, -> extractTyVarsFromTy, TyVar, SigmaType(..) +> tyVarsOfType, TyVar, SigmaType(..) > IF_ATTACK_PRAGMAS(COMMA cmpTyVar) > ) > import Literal ( Literal ) -- for Eq Literal @@ -158,8 +158,8 @@ but l is guranteed to be finite so we choose that one. > Let (Rec bs) e -> foldr freeBind (free e tvs) bs > SCC l e -> free e tvs > -> freeId id tvs = extractTyVarsFromTy (idType id) `union` tvs -> freeTy t tvs = extractTyVarsFromTy t `union` tvs +> freeId id tvs = tyVarsOfType (idType id) `union` tvs +> freeTy t tvs = tyVarsOfType t `union` tvs > freeBind (v,e) tvs = freeId v (free e tvs) > freeAtom (VarArg (DefArgExpr e)) tvs = free e tvs diff --git a/ghc/compiler/parser/printtree.c b/ghc/compiler/parser/printtree.c index a5056ef635..45c89be0ad 100644 --- a/ghc/compiler/parser/printtree.c +++ b/ghc/compiler/parser/printtree.c @@ -500,7 +500,7 @@ prbind(b) PUTTAG('e'); printf("#%lu\t",gibindline(b)); pid(gibindfile(b)); - pid(gibindmod(b)); + pid(gibindimod(b)); /* plist(pentid,giebindexp(b)); ??? */ /* prbind(giebinddef(b)); ???? */ break; @@ -661,12 +661,12 @@ ppragma(p) break; case iinst_simpl_pragma: PUTTAGSTR("Pis"); - pid(gprag_imod_simpl(p)); - ppragma(gprag_dfun_simpl(p)); +/* pid(gprag_imod_simpl(p)); +*/ ppragma(gprag_dfun_simpl(p)); break; case iinst_const_pragma: PUTTAGSTR("Pic"); - pid(gprag_imod_const(p)); - ppragma(gprag_dfun_const(p)); +/* pid(gprag_imod_const(p)); +*/ ppragma(gprag_dfun_const(p)); plist(ppragma, gprag_constms(p)); break; diff --git a/ghc/compiler/prelude/TysPrim.lhs b/ghc/compiler/prelude/TysPrim.lhs index afc81b93b3..c16c6b87b5 100644 --- a/ghc/compiler/prelude/TysPrim.lhs +++ b/ghc/compiler/prelude/TysPrim.lhs @@ -20,14 +20,14 @@ import PrimRep ( PrimRep(..) ) -- getPrimRepInfo uses PrimRep repn import TyCon ( mkPrimTyCon, mkDataTyCon, ConsVisible(..), NewOrData(..) ) import TyVar ( GenTyVar(..), alphaTyVars ) -import Type ( applyTyCon, mkTyVarTy ) +import Type ( applyTyCon, mkTyVarTys ) import Usage ( usageOmega ) import Unique \end{code} \begin{code} -alphaTys = map mkTyVarTy alphaTyVars +alphaTys = mkTyVarTys alphaTyVars (alphaTy:betaTy:gammaTy:deltaTy:_) = alphaTys \end{code} diff --git a/ghc/compiler/profiling/SCCauto.lhs b/ghc/compiler/profiling/SCCauto.lhs index ba3da63cb8..eb8f1430ad 100644 --- a/ghc/compiler/profiling/SCCauto.lhs +++ b/ghc/compiler/profiling/SCCauto.lhs @@ -73,7 +73,7 @@ addAutoCostCentres mod_name binds scc_rhs rhs = let - (usevars, tyvars, vars, body) = digForLambdas rhs + (usevars, tyvars, vars, body) = collectBinders rhs in case body of SCC _ _ -> rhs -- leave it diff --git a/ghc/compiler/simplCore/AnalFBWW.lhs b/ghc/compiler/simplCore/AnalFBWW.lhs index c2b8f8d569..7e456079cc 100644 --- a/ghc/compiler/simplCore/AnalFBWW.lhs +++ b/ghc/compiler/simplCore/AnalFBWW.lhs @@ -180,7 +180,7 @@ analBind (NonRec (v,bnd) e) env = analBind (Rec binds) env = let first_set = [ (v,IsFB (FBType [FBBadConsum | _ <- args ] FBGoodProd)) | ((v,_),e) <- binds, - (_,_,args,_) <- [digForLambdas e]] + (_,_,args,_) <- [collectBinders e]] env' = delManyFromIdEnv env (map (fst.fst) binds) in growIdEnvList env' (fixpoint 0 binds env' first_set) diff --git a/ghc/compiler/simplCore/ConFold.lhs b/ghc/compiler/simplCore/ConFold.lhs index 0a128aed9b..c508cf59fd 100644 --- a/ghc/compiler/simplCore/ConFold.lhs +++ b/ghc/compiler/simplCore/ConFold.lhs @@ -1,5 +1,5 @@ % -% (c) The GRASP/AQUA Project, Glasgow University, 1992-1995 +% (c) The GRASP/AQUA Project, Glasgow University, 1992-1996 % \section[ConFold]{Constant Folder} @@ -12,22 +12,22 @@ ToDo: module ConFold ( completePrim ) where -import SimplEnv -import SimplMonad +import Ubiq{-uitous-} -import PrelInfo ( trueDataCon, falseDataCon, PrimOp(..), PrimRep - IF_ATTACK_PRAGMAS(COMMA tagOf_PrimOp) - IF_ATTACK_PRAGMAS(COMMA pprPrimOp) - ) +import CoreSyn +import CoreUnfold ( UnfoldingDetails(..), FormSummary(..) ) +import Id ( idType ) import Literal ( mkMachInt, mkMachWord, Literal(..) ) -import Id ( Id, idType ) -import Maybes ( Maybe(..) ) -import Util +import MagicUFs ( MagicUnfoldingFun ) +import PrelInfo ( trueDataCon, falseDataCon ) +import PrimOp ( PrimOp(..) ) +import SimplEnv +import SimplMonad \end{code} \begin{code} completePrim :: SimplEnv - -> PrimOp -> [OutType] -> [OutAtom] + -> PrimOp -> [OutArg] -> SmplM OutExpr \end{code} @@ -86,58 +86,57 @@ NB: If we ever do case-floating, we have an extra worry: The second case must never be floated outside of the first! \begin{code} -completePrim env SeqOp [ty] [LitArg lit] +completePrim env SeqOp [TyArg ty, LitArg lit] = returnSmpl (Lit (mkMachInt 1)) -completePrim env op@SeqOp tys@[ty] args@[VarArg var] +completePrim env op@SeqOp args@[TyArg ty, VarArg var] = case (lookupUnfolding env var) of - NoUnfoldingDetails -> give_up - LitForm _ -> hooray - OtherLitForm _ -> hooray - ConForm _ _ _ -> hooray - OtherConForm _ -> hooray + NoUnfoldingDetails -> give_up + LitForm _ -> hooray + OtherLitForm _ -> hooray + ConForm _ _ -> hooray + OtherConForm _ -> hooray GenForm _ WhnfForm _ _ -> hooray - _ -> give_up + _ -> give_up where - give_up = returnSmpl (Prim op tys args) - hooray = returnSmpl (Lit (mkMachInt 1)) + give_up = returnSmpl (Prim op args) + hooray = returnSmpl (Lit (mkMachInt 1)) \end{code} \begin{code} -completePrim env op tys args +completePrim env op args = case args of - [LitArg (MachChar char_lit)] -> oneCharLit op char_lit - [LitArg (MachInt int_lit signed)] -> (if signed then oneIntLit else oneWordLit) - op int_lit - [LitArg (MachFloat float_lit)] -> oneFloatLit op float_lit - [LitArg (MachDouble double_lit)] -> oneDoubleLit op double_lit - [LitArg other_lit] -> oneLit op other_lit - - [LitArg (MachChar char_lit1), - LitArg (MachChar char_lit2)] -> twoCharLits op char_lit1 char_lit2 + [LitArg (MachChar char_lit)] -> oneCharLit op char_lit + [LitArg (MachInt int_lit signed)] -> (if signed then oneIntLit else oneWordLit) + op int_lit + [LitArg (MachFloat float_lit)] -> oneFloatLit op float_lit + [LitArg (MachDouble double_lit)] -> oneDoubleLit op double_lit + [LitArg other_lit] -> oneLit op other_lit - [LitArg (MachInt int_lit1 True), -- both *signed* literals - LitArg (MachInt int_lit2 True)] -> twoIntLits op int_lit1 int_lit2 + [LitArg (MachChar char_lit1), + LitArg (MachChar char_lit2)] -> twoCharLits op char_lit1 char_lit2 - [LitArg (MachInt int_lit1 False), -- both *unsigned* literals - LitArg (MachInt int_lit2 False)] -> twoWordLits op int_lit1 int_lit2 + [LitArg (MachInt int_lit1 True), -- both *signed* literals + LitArg (MachInt int_lit2 True)] -> twoIntLits op int_lit1 int_lit2 - [LitArg (MachInt int_lit1 False), -- unsigned+signed (shift ops) - LitArg (MachInt int_lit2 True)] -> oneWordOneIntLit op int_lit1 int_lit2 + [LitArg (MachInt int_lit1 False), -- both *unsigned* literals + LitArg (MachInt int_lit2 False)] -> twoWordLits op int_lit1 int_lit2 - [LitArg (MachFloat float_lit1), - LitArg (MachFloat float_lit2)] -> twoFloatLits op float_lit1 float_lit2 + [LitArg (MachInt int_lit1 False), -- unsigned+signed (shift ops) + LitArg (MachInt int_lit2 True)] -> oneWordOneIntLit op int_lit1 int_lit2 - [LitArg (MachDouble double_lit1), - LitArg (MachDouble double_lit2)] -> twoDoubleLits op double_lit1 double_lit2 + [LitArg (MachFloat float_lit1), + LitArg (MachFloat float_lit2)] -> twoFloatLits op float_lit1 float_lit2 - [LitArg lit, VarArg var] -> litVar op lit var - [VarArg var, LitArg lit] -> litVar op lit var + [LitArg (MachDouble double_lit1), + LitArg (MachDouble double_lit2)] -> twoDoubleLits op double_lit1 double_lit2 - other -> give_up + [LitArg lit, VarArg var] -> litVar op lit var + [VarArg var, LitArg lit] -> litVar op lit var + other -> give_up where - give_up = returnSmpl (Prim op tys args) + give_up = returnSmpl (Prim op args) return_char c = returnSmpl (Lit (MachChar c)) return_int i = returnSmpl (Lit (mkMachInt i)) @@ -157,9 +156,7 @@ completePrim env op tys args (PrimAlts [(lit,val_if_eq)] (BindDefault unused_binder val_if_neq)) in --- pprTrace "return_prim_case:" (ppr PprDebug result) ( returnSmpl result --- ) --------- Ints -------------- oneIntLit IntNegOp i = return_int (-i) @@ -188,7 +185,7 @@ completePrim env op tys args twoIntLits IntLtOp i1 i2 = return_bool (i1 < i2) twoIntLits IntLeOp i1 i2 = return_bool (i1 <= i2) -- ToDo: something for integer-shift ops? - twoIntLits _ _ _ = {-trace "twoIntLits: giving up"-} give_up + twoIntLits _ _ _ = give_up twoWordLits WordGtOp w1 w2 = return_bool (w1 > w2) twoWordLits WordGeOp w1 w2 = return_bool (w1 >= w2) @@ -197,10 +194,10 @@ completePrim env op tys args twoWordLits WordLtOp w1 w2 = return_bool (w1 < w2) twoWordLits WordLeOp w1 w2 = return_bool (w1 <= w2) -- ToDo: something for AndOp, OrOp? - twoWordLits _ _ _ = {-trace "twoWordLits: giving up"-} give_up + twoWordLits _ _ _ = give_up -- ToDo: something for shifts - oneWordOneIntLit _ _ _ = {-trace "oneWordOneIntLit: giving up"-} give_up + oneWordOneIntLit _ _ _ = give_up --------- Floats -------------- oneFloatLit FloatNegOp f = return_float (-f) @@ -220,7 +217,7 @@ completePrim env op tys args #else -- hard to do all that in Rationals ?? (WDP 94/10) ToDo #endif - oneFloatLit _ _ = {-trace "oneFloatLits: giving up"-} give_up + oneFloatLit _ _ = give_up twoFloatLits FloatGtOp f1 f2 = return_bool (f1 > f2) twoFloatLits FloatGeOp f1 f2 = return_bool (f1 >= f2) @@ -232,32 +229,11 @@ completePrim env op tys args twoFloatLits FloatSubOp f1 f2 = return_float (f1 - f2) twoFloatLits FloatMulOp f1 f2 = return_float (f1 * f2) twoFloatLits FloatDivOp f1 f2 | f2 /= 0 = return_float (f1 / f2) -#if __GLASGOW_HASKELL__ <= 22 - twoFloatLits FloatPowerOp f1 f2 = return_float (f1 ** f2) -#else - -- hard to do all that in Rationals ?? (WDP 94/10) ToDo -#endif - twoFloatLits _ _ _ = {-trace "twoFloatLits: giving up"-} give_up + twoFloatLits _ _ _ = give_up --------- Doubles -------------- oneDoubleLit DoubleNegOp d = return_double (-d) -#if __GLASGOW_HASKELL__ <= 22 - oneDoubleLit DoubleExpOp d = return_double (exp d) - oneDoubleLit DoubleLogOp d = return_double (log d) - oneDoubleLit DoubleSqrtOp d = return_double (sqrt d) - oneDoubleLit DoubleSinOp d = return_double (sin d) - oneDoubleLit DoubleCosOp d = return_double (cos d) - oneDoubleLit DoubleTanOp d = return_double (tan d) - oneDoubleLit DoubleAsinOp d = return_double (asin d) - oneDoubleLit DoubleAcosOp d = return_double (acos d) - oneDoubleLit DoubleAtanOp d = return_double (atan d) - oneDoubleLit DoubleSinhOp d = return_double (sinh d) - oneDoubleLit DoubleCoshOp d = return_double (cosh d) - oneDoubleLit DoubleTanhOp d = return_double (tanh d) -#else - -- hard to do all that in Rationals ?? (WDP 94/10) ToDo -#endif - oneDoubleLit _ _ = {-trace "oneDoubleLit: giving up"-} give_up + oneDoubleLit _ _ = give_up twoDoubleLits DoubleGtOp d1 d2 = return_bool (d1 > d2) twoDoubleLits DoubleGeOp d1 d2 = return_bool (d1 >= d2) @@ -269,16 +245,11 @@ completePrim env op tys args twoDoubleLits DoubleSubOp d1 d2 = return_double (d1 - d2) twoDoubleLits DoubleMulOp d1 d2 = return_double (d1 * d2) twoDoubleLits DoubleDivOp d1 d2 | d2 /= 0 = return_double (d1 / d2) -#if __GLASGOW_HASKELL__ <= 22 - twoDoubleLits DoublePowerOp d1 d2 = return_double (d1 ** d2) -#else - -- hard to do all that in Rationals ?? (WDP 94/10) ToDo -#endif - twoDoubleLits _ _ _ = {-trace "twoDoubleLits: giving up"-} give_up + twoDoubleLits _ _ _ = give_up --------- Characters -------------- oneCharLit OrdOp c = return_int (fromInt (ord c)) - oneCharLit _ _ = {-trace "oneCharLIt: giving up"-} give_up + oneCharLit _ _ = give_up twoCharLits CharGtOp c1 c2 = return_bool (c1 > c2) twoCharLits CharGeOp c1 c2 = return_bool (c1 >= c2) @@ -286,7 +257,7 @@ completePrim env op tys args twoCharLits CharNeOp c1 c2 = return_bool (c1 /= c2) twoCharLits CharLtOp c1 c2 = return_bool (c1 < c2) twoCharLits CharLeOp c1 c2 = return_bool (c1 <= c2) - twoCharLits _ _ _ = {-trace "twoCharLits: giving up"-} give_up + twoCharLits _ _ _ = give_up --------- Miscellaneous -------------- oneLit Addr2IntOp (MachAddr i) = return_int i @@ -319,6 +290,6 @@ completePrim env op tys args litVar other_op lit var = give_up -trueVal = Con trueDataCon [] [] -falseVal = Con falseDataCon [] [] +trueVal = Con trueDataCon [] +falseVal = Con falseDataCon [] \end{code} diff --git a/ghc/compiler/simplCore/FloatIn.lhs b/ghc/compiler/simplCore/FloatIn.lhs index c8b25177f5..27b6c08f14 100644 --- a/ghc/compiler/simplCore/FloatIn.lhs +++ b/ghc/compiler/simplCore/FloatIn.lhs @@ -20,11 +20,16 @@ module FloatIn ( -- and to make the interface self-sufficient... ) where +import Ubiq{-uitous-} + import AnnCoreSyn +import CoreSyn import FreeVars -import UniqSet -import Util +import Id ( emptyIdSet, unionIdSets, unionManyIdSets, + elementOfIdSet, IdSet(..) + ) +import Util ( panic ) \end{code} Top-level interface function, @floatInwards@. Note that we do not @@ -113,7 +118,7 @@ the closure for a is not built. %************************************************************************ \begin{code} -type FreeVarsSet = UniqSet Id +type FreeVarsSet = IdSet type FloatingBinds = [(CoreBinding, FreeVarsSet)] -- In dependency order (outermost first) @@ -127,23 +132,26 @@ fiExpr :: FloatingBinds -- binds we're trying to drop -> CoreExprWithFVs -- input expr -> CoreExpr -- result -fiExpr to_drop (_,AnnCoVar v) = mkCoLets' to_drop (Var v) +fiExpr to_drop (_,AnnVar v) = mkCoLets' to_drop (Var v) -fiExpr to_drop (_,AnnCoLit k) = mkCoLets' to_drop (Lit k) +fiExpr to_drop (_,AnnLit k) = mkCoLets' to_drop (Lit k) -fiExpr to_drop (_,AnnCoCon c tys atoms) - = mkCoLets' to_drop (Con c tys atoms) +fiExpr to_drop (_,AnnCon c atoms) + = mkCoLets' to_drop (Con c atoms) -fiExpr to_drop (_,AnnCoPrim c tys atoms) - = mkCoLets' to_drop (Prim c tys atoms) +fiExpr to_drop (_,AnnPrim c atoms) + = mkCoLets' to_drop (Prim c atoms) \end{code} Here we are not floating inside lambda (type lambdas are OK): \begin{code} -fiExpr to_drop (_,AnnCoLam binder body) - = mkCoLets' to_drop (Lam binder (fiExpr [] body)) +fiExpr to_drop (_,AnnLam (UsageBinder binder) body) + = panic "FloatIn.fiExpr:AnnLam UsageBinder" + +fiExpr to_drop (_,AnnLam b@(ValBinder binder) body) + = mkCoLets' to_drop (Lam b (fiExpr [] body)) -fiExpr to_drop (_,AnnCoTyLam tyvar body) +fiExpr to_drop (_,AnnLam b@(TyBinder tyvar) body) | whnf body -- we do not float into type lambdas if they are followed by -- a whnf (actually we check for lambdas and constructors). @@ -157,28 +165,30 @@ fiExpr to_drop (_,AnnCoTyLam tyvar body) -- let f = /\t -> let v = ... in \a -> ... -- which is bad as now f is an updatable closure (update PAP) -- and has arity 0. This example comes from cichelli. - = mkCoLets' to_drop (CoTyLam tyvar (fiExpr [] body)) + + = mkCoLets' to_drop (Lam b (fiExpr [] body)) | otherwise - = CoTyLam tyvar (fiExpr to_drop body) + = Lam b (fiExpr to_drop body) where whnf :: CoreExprWithFVs -> Bool - whnf (_,AnnCoLit _) = True - whnf (_,AnnCoCon _ _ _) = True - whnf (_,AnnCoLam _ _) = True - whnf (_,AnnCoTyLam _ e) = whnf e - whnf (_,AnnCoSCC _ e) = whnf e - whnf _ = False + + whnf (_,AnnLit _) = True + whnf (_,AnnCon _ _) = True + whnf (_,AnnLam (ValBinder _) _) = True + whnf (_,AnnLam _ e) = whnf e + whnf (_,AnnSCC _ e) = whnf e + whnf _ = False \end{code} Applications: we could float inside applications, but it's probably not worth it (a purely practical choice, hunch- [not experience-] based). \begin{code} -fiExpr to_drop (_,AnnCoApp fun atom) - = mkCoLets' to_drop (App (fiExpr [] fun) atom) - -fiExpr to_drop (_,AnnCoTyApp expr ty) - = CoTyApp (fiExpr to_drop expr) ty +fiExpr to_drop (_,AnnApp fun arg) + | isValArg arg + = mkCoLets' to_drop (App (fiExpr [] fun) arg) + | otherwise + = App (fiExpr to_drop fun) arg \end{code} We don't float lets inwards past an SCC. @@ -187,7 +197,7 @@ ToDo: SCC: {\em should} keep info on current cc, and when passing one, if it is not the same, annotate all lets in binds with current cc, change current cc to the new one and float binds into expr. \begin{code} -fiExpr to_drop (_, AnnCoSCC cc expr) +fiExpr to_drop (_, AnnSCC cc expr) = mkCoLets' to_drop (SCC cc (fiExpr [] expr)) \end{code} @@ -214,7 +224,7 @@ things to drop in the outer let's body, and let nature take its course. \begin{code} -fiExpr to_drop (_,AnnCoLet (AnnCoNonRec id rhs) body) +fiExpr to_drop (_,AnnLet (AnnNonRec id rhs) body) = fiExpr new_to_drop body where rhs_fvs = freeVarsOf rhs @@ -228,9 +238,9 @@ fiExpr to_drop (_,AnnCoLet (AnnCoNonRec id rhs) body) -- Push rhs_binds into the right hand side of the binding rhs' = fiExpr rhs_binds rhs - rhs_fvs' = rhs_fvs `unionUniqSets` (floatedBindsFVs rhs_binds) + rhs_fvs' = rhs_fvs `unionIdSets` floatedBindsFVs rhs_binds -fiExpr to_drop (_,AnnCoLet (AnnCoRec bindings) body) +fiExpr to_drop (_,AnnLet (AnnRec bindings) body) = fiExpr new_to_drop body where (binders, rhss) = unzip bindings @@ -248,8 +258,8 @@ fiExpr to_drop (_,AnnCoLet (AnnCoRec bindings) body) -- the bindings used both in rhs and body or in more than one rhs shared_binds - rhs_fvs' = unionUniqSets (unionManyUniqSets rhss_fvs) - (unionManyUniqSets (map floatedBindsFVs rhss_binds)) + rhs_fvs' = unionIdSets (unionManyIdSets rhss_fvs) + (unionManyIdSets (map floatedBindsFVs rhss_binds)) -- Push rhs_binds into the right hand side of the binding fi_bind :: [FloatingBinds] -- one per "drop pt" conjured w/ fvs_of_rhss @@ -265,7 +275,7 @@ bindings are: (a)~inside the scrutinee, (b)~inside one of the alternatives/default [default FVs always {\em first}!]. \begin{code} -fiExpr to_drop (_, AnnCoCase scrut alts) +fiExpr to_drop (_, AnnCase scrut alts) = let fvs_scrut = freeVarsOf scrut drop_pts_fvs = fvs_scrut : (get_fvs_from_deflt_and_alts alts) @@ -279,30 +289,30 @@ fiExpr to_drop (_, AnnCoCase scrut alts) ---------------------------- -- pin default FVs on first! -- - get_fvs_from_deflt_and_alts (AnnCoAlgAlts alts deflt) + get_fvs_from_deflt_and_alts (AnnAlgAlts alts deflt) = get_deflt_fvs deflt : [ freeVarsOf rhs | (_, _, rhs) <- alts ] - get_fvs_from_deflt_and_alts (AnnCoPrimAlts alts deflt) + get_fvs_from_deflt_and_alts (AnnPrimAlts alts deflt) = get_deflt_fvs deflt : [ freeVarsOf rhs | (_, rhs) <- alts] - get_deflt_fvs AnnCoNoDefault = emptyUniqSet - get_deflt_fvs (AnnCoBindDefault b rhs) = freeVarsOf rhs + get_deflt_fvs AnnNoDefault = emptyIdSet + get_deflt_fvs (AnnBindDefault b rhs) = freeVarsOf rhs ---------------------------- - fi_alts to_drop_deflt to_drop_alts (AnnCoAlgAlts alts deflt) + fi_alts to_drop_deflt to_drop_alts (AnnAlgAlts alts deflt) = AlgAlts [ (con, params, fiExpr to_drop rhs) | ((con, params, rhs), to_drop) <- alts `zip` to_drop_alts ] (fi_default to_drop_deflt deflt) - fi_alts to_drop_deflt to_drop_alts (AnnCoPrimAlts alts deflt) + fi_alts to_drop_deflt to_drop_alts (AnnPrimAlts alts deflt) = PrimAlts [ (lit, fiExpr to_drop rhs) | ((lit, rhs), to_drop) <- alts `zip` to_drop_alts ] (fi_default to_drop_deflt deflt) - fi_default to_drop AnnCoNoDefault = NoDefault - fi_default to_drop (AnnCoBindDefault b e) = BindDefault b (fiExpr to_drop e) + fi_default to_drop AnnNoDefault = NoDefault + fi_default to_drop (AnnBindDefault b e) = BindDefault b (fiExpr to_drop e) \end{code} %************************************************************************ @@ -341,7 +351,7 @@ sepBindsByDropPoint drop_pts [] sepBindsByDropPoint drop_pts floaters = let (per_drop_pt, must_stay_here, _) - --= sep drop_pts emptyUniqSet{-fvs of prev drop_pts-} floaters + --= sep drop_pts emptyIdSet{-fvs of prev drop_pts-} floaters = split' drop_pts floaters [] empty_boxes empty_boxes = take (length drop_pts) (repeat []) @@ -353,16 +363,16 @@ sepBindsByDropPoint drop_pts floaters -- only in a or unused split' (a:as) (bind:binds) mult_branch (drop_box_a:drop_boxes) - | all (\b -> {-b `elementOfUniqSet` a &&-} - not (b `elementOfUniqSet` (unionManyUniqSets as))) + | all (\b -> {-b `elementOfIdSet` a &&-} + not (b `elementOfIdSet` (unionManyIdSets as))) (bindersOf (fst bind)) = split' (a':as) binds mult_branch ((bind:drop_box_a):drop_boxes) where - a' = a `unionUniqSets` fvsOfBind bind + a' = a `unionIdSets` fvsOfBind bind -- not in a split' (a:as) (bind:binds) mult_branch (drop_box_a:drop_boxes) - | all (\b -> not (b `elementOfUniqSet` a)) (bindersOf (fst bind)) + | all (\b -> not (b `elementOfIdSet` a)) (bindersOf (fst bind)) = split' (a:as') binds mult_branch' (drop_box_a:drop_boxes') where (drop_boxes',mult_branch',as') = split' as [bind] mult_branch drop_boxes @@ -371,13 +381,13 @@ sepBindsByDropPoint drop_pts floaters split' aas@(a:as) (bind:binds) mult_branch drop_boxes = split' aas' binds (bind : mult_branch) drop_boxes where - aas' = map (unionUniqSets (fvsOfBind bind)) aas + aas' = map (unionIdSets (fvsOfBind bind)) aas ------------------------- fvsOfBind (_,fvs) = fvs --floatedBindsFVs :: -floatedBindsFVs binds = foldr unionUniqSets emptyUniqSet (map snd binds) +floatedBindsFVs binds = unionManyIdSets (map snd binds) --mkCoLets' :: [FloatingBinds] -> CoreExpr -> CoreExpr mkCoLets' to_drop e = mkCoLetsNoUnboxed (reverse (map fst to_drop)) e diff --git a/ghc/compiler/simplCore/FloatOut.lhs b/ghc/compiler/simplCore/FloatOut.lhs index 000ed33dd3..d65112ac0b 100644 --- a/ghc/compiler/simplCore/FloatOut.lhs +++ b/ghc/compiler/simplCore/FloatOut.lhs @@ -10,27 +10,40 @@ module FloatOut ( floatOutwards ) where -import Literal ( Literal(..) ) -import CmdLineOpts ( GlobalSwitch(..) ) -import CostCentre ( dupifyCC, CostCentre ) -import SetLevels -import Id ( eqId ) -import Maybes ( Maybe(..), catMaybes, maybeToBool ) -import UniqSupply -import Util +import Ubiq{-uitous-} + +import CoreSyn + +import CmdLineOpts ( opt_D_verbose_core2core, opt_D_simplifier_stats ) +import CostCentre ( dupifyCC ) +import Id ( nullIdEnv, addOneToIdEnv, growIdEnvList, IdEnv(..), + GenId{-instance Outputable-} + ) +import Outputable ( Outputable(..){-instance (,)-} ) +import PprCore ( GenCoreBinding{-instance-} ) +import PprStyle ( PprStyle(..) ) +import PprType -- too lazy to type in all the instances +import Pretty ( ppInt, ppStr, ppBesides, ppAboves ) +import SetLevels -- all of it +import TyVar ( GenTyVar{-instance Eq-} ) +import Unique ( Unique{-instance Eq-} ) +import Usage ( UVar(..) ) +import Util ( pprTrace, panic ) \end{code} Random comments ~~~~~~~~~~~~~~~ -At the moment we never float a binding out to between two adjacent lambdas. For -example: + +At the moment we never float a binding out to between two adjacent +lambdas. For example: + @ \x y -> let t = x+x in ... ===> \x -> let t = x+x in \y -> ... @ -Reason: this is less efficient in the case where the original lambda is -never partially applied. +Reason: this is less efficient in the case where the original lambda +is never partially applied. But there's a case I've seen where this might not be true. Consider: @ @@ -50,19 +63,19 @@ which might usefully be separated to @ Well, maybe. We don't do this at the moment. - \begin{code} -type LevelledExpr = GenCoreExpr (Id, Level) Id -type LevelledBind = GenCoreBinding (Id, Level) Id +type LevelledExpr = GenCoreExpr (Id, Level) Id TyVar UVar +type LevelledBind = GenCoreBinding (Id, Level) Id TyVar UVar type FloatingBind = (Level, Floater) type FloatingBinds = [FloatingBind] -data Floater = LetFloater CoreBinding - - | CaseFloater (CoreExpr -> CoreExpr) - -- Give me a right-hand side of the - -- (usually single) alternative, and - -- I'll build the case +data Floater + = LetFloater CoreBinding + | CaseFloater (CoreExpr -> CoreExpr) + -- A CoreExpr with a hole in it: + -- "Give me a right-hand side of the + -- (usually single) alternative, and + -- I'll build the case..." \end{code} %************************************************************************ @@ -72,22 +85,20 @@ data Floater = LetFloater CoreBinding %************************************************************************ \begin{code} -floatOutwards :: (GlobalSwitch -> Bool) -- access to all global cmd-line opts - -> UniqSupply - -> [CoreBinding] - -> [CoreBinding] +floatOutwards :: UniqSupply -> [CoreBinding] -> [CoreBinding] -floatOutwards sw_chker us pgm - = case (setLevels pgm sw_chker us) of { annotated_w_levels -> +floatOutwards us pgm + = case (setLevels pgm us) of { annotated_w_levels -> - case unzip (map (floatTopBind sw_chker) annotated_w_levels) + case (unzip (map floatTopBind annotated_w_levels)) of { (fss, final_toplev_binds_s) -> - (if sw_chker D_verbose_core2core - then pprTrace "Levels added:\n" (ppr PprDebug annotated_w_levels) + (if opt_D_verbose_core2core + then pprTrace "Levels added:\n" + (ppAboves (map (ppr PprDebug) annotated_w_levels)) else id ) - ( if not (sw_chker D_simplifier_stats) then + ( if not (opt_D_simplifier_stats) then id else let @@ -101,13 +112,13 @@ floatOutwards sw_chker us pgm concat final_toplev_binds_s }} -floatTopBind sw bind@(NonRec _ _) - = case (floatBind sw nullIdEnv tOP_LEVEL bind) of { (fs, floats, bind', _) -> +floatTopBind bind@(NonRec _ _) + = case (floatBind nullIdEnv tOP_LEVEL bind) of { (fs, floats, bind', _) -> (fs, floatsToBinds floats ++ [bind']) } -floatTopBind sw bind@(Rec _) - = case (floatBind sw nullIdEnv tOP_LEVEL bind) of { (fs, floats, Rec pairs', _) -> +floatTopBind bind@(Rec _) + = case (floatBind nullIdEnv tOP_LEVEL bind) of { (fs, floats, Rec pairs', _) -> -- Actually floats will be empty --false:ASSERT(null floats) (fs, [Rec (floatsToBindPairs floats ++ pairs')]) @@ -122,22 +133,23 @@ floatTopBind sw bind@(Rec _) \begin{code} -floatBind :: (GlobalSwitch -> Bool) - -> IdEnv Level +floatBind :: IdEnv Level -> Level -> LevelledBind -> (FloatStats, FloatingBinds, CoreBinding, IdEnv Level) -floatBind sw env lvl (NonRec (name,level) rhs) - = case (floatExpr sw env level rhs) of { (fs, rhs_floats, rhs') -> +floatBind env lvl (NonRec (name,level) rhs) + = case (floatExpr env level rhs) of { (fs, rhs_floats, rhs') -> -- A good dumping point - case (partitionByMajorLevel level rhs_floats) of { (rhs_floats', heres) -> + case (partitionByMajorLevel level rhs_floats) of { (rhs_floats', heres) -> - (fs, rhs_floats',NonRec name (install heres rhs'), addOneToIdEnv env name level) + (fs, rhs_floats', + NonRec name (install heres rhs'), + addOneToIdEnv env name level) }} -floatBind sw env lvl bind@(Rec pairs) +floatBind env lvl bind@(Rec pairs) = case (unzip3 (map do_pair pairs)) of { (fss, rhss_floats, new_pairs) -> if not (isTopLvl bind_level) then @@ -171,10 +183,10 @@ floatBind sw env lvl bind@(Rec pairs) bind_level = getBindLevel bind do_pair ((name, level), rhs) - = case (floatExpr sw new_env level rhs) of { (fs, rhs_floats, rhs') -> + = case (floatExpr new_env level rhs) of { (fs, rhs_floats, rhs') -> -- A good dumping point - case (partitionByMajorLevel level rhs_floats) of { (rhs_floats', heres) -> + case (partitionByMajorLevel level rhs_floats) of { (rhs_floats', heres) -> (fs, rhs_floats', (name, install heres rhs')) }} @@ -187,55 +199,51 @@ floatBind sw env lvl bind@(Rec pairs) %************************************************************************ \begin{code} -floatExpr :: (GlobalSwitch -> Bool) - -> IdEnv Level +floatExpr :: IdEnv Level -> Level -> LevelledExpr -> (FloatStats, FloatingBinds, CoreExpr) -floatExpr sw env _ (Var v) = (zero_stats, [], Var v) - -floatExpr sw env _ (Lit l) = (zero_stats, [], Lit l) - -floatExpr sw env _ (Prim op ty as) = (zero_stats, [], Prim op ty as) -floatExpr sw env _ (Con con ty as) = (zero_stats, [], Con con ty as) - -floatExpr sw env lvl (App e a) - = case (floatExpr sw env lvl e) of { (fs, floating_defns, e') -> +floatExpr env _ (Var v) = (zero_stats, [], Var v) +floatExpr env _ (Lit l) = (zero_stats, [], Lit l) +floatExpr env _ (Prim op as) = (zero_stats, [], Prim op as) +floatExpr env _ (Con con as) = (zero_stats, [], Con con as) + +floatExpr env lvl (App e a) + = case (floatExpr env lvl e) of { (fs, floating_defns, e') -> (fs, floating_defns, App e' a) } -floatExpr sw env lvl (CoTyApp e ty) - = case (floatExpr sw env lvl e) of { (fs, floating_defns, e') -> - (fs, floating_defns, CoTyApp e' ty) } +floatExpr env lvl (Lam (UsageBinder _) e) + = panic "FloatOut.floatExpr: Lam UsageBinder" -floatExpr sw env lvl (CoTyLam tv e) +floatExpr env lvl (Lam (TyBinder tv) e) = let incd_lvl = incMinorLvl lvl in - case (floatExpr sw env incd_lvl e) of { (fs, floats, e') -> + case (floatExpr env incd_lvl e) of { (fs, floats, e') -> -- Dump any bindings which absolutely cannot go any further case (partitionByLevel incd_lvl floats) of { (floats', heres) -> - (fs, floats', CoTyLam tv (install heres e')) + (fs, floats', Lam (TyBinder tv) (install heres e')) }} -floatExpr sw env lvl (Lam (arg,incd_lvl) rhs) +floatExpr env lvl (Lam (ValBinder (arg,incd_lvl)) rhs) = let new_env = addOneToIdEnv env arg incd_lvl in - case (floatExpr sw new_env incd_lvl rhs) of { (fs, floats, rhs') -> + case (floatExpr new_env incd_lvl rhs) of { (fs, floats, rhs') -> -- Dump any bindings which absolutely cannot go any further case (partitionByLevel incd_lvl floats) of { (floats', heres) -> (add_to_stats fs floats', floats', - Lam args' (install heres rhs')) + Lam (ValBinder arg) (install heres rhs')) }} -floatExpr sw env lvl (SCC cc expr) - = case (floatExpr sw env lvl expr) of { (fs, floating_defns, expr') -> +floatExpr env lvl (SCC cc expr) + = case (floatExpr env lvl expr) of { (fs, floating_defns, expr') -> let -- annotate bindings floated outwards past an scc expression -- with the cc. We mark that cc as "duplicated", though. @@ -257,17 +265,16 @@ floatExpr sw env lvl (SCC cc expr) ann_bind (CaseFloater fn) = CaseFloater ( \ rhs -> SCC dupd_cc (fn rhs) ) - ann_rhs (Lam arg e) = Lam arg (ann_rhs e) - ann_rhs (CoTyLam tv e) = CoTyLam tv (ann_rhs e) - ann_rhs rhs@(Con _ _ _)= rhs -- no point in scc'ing WHNF data - ann_rhs rhs = SCC dupd_cc rhs + ann_rhs (Lam arg e) = Lam arg (ann_rhs e) + ann_rhs rhs@(Con _ _) = rhs -- no point in scc'ing WHNF data + ann_rhs rhs = SCC dupd_cc rhs -- Note: Nested SCC's are preserved for the benefit of -- cost centre stack profiling (Durham) -floatExpr sw env lvl (Let bind body) - = case (floatBind sw env lvl bind) of { (fsb, rhs_floats, bind', new_env) -> - case (floatExpr sw new_env lvl body) of { (fse, body_floats, body') -> +floatExpr env lvl (Let bind body) + = case (floatBind env lvl bind) of { (fsb, rhs_floats, bind', new_env) -> + case (floatExpr new_env lvl body) of { (fse, body_floats, body') -> (add_stats fsb fse, rhs_floats ++ [(bind_lvl, LetFloater bind')] ++ body_floats, body') @@ -275,12 +282,14 @@ floatExpr sw env lvl (Let bind body) where bind_lvl = getBindLevel bind -floatExpr sw env lvl (Case scrut alts) - = case (floatExpr sw env lvl scrut) of { (fse, fde, scrut') -> +floatExpr env lvl (Case scrut alts) + = case (floatExpr env lvl scrut) of { (fse, fde, scrut') -> case (scrut', float_alts alts) of - -{- CASE-FLOATING DROPPED FOR NOW. (SLPJ 7/2/94) + (_, (fsa, fda, alts')) -> + (add_stats fse fsa, fda ++ fde, Case scrut' alts') + } + {- OLD CASE-FLOATING CODE: DROPPED FOR NOW. (SLPJ 7/2/94) (Var scrut_var, (fda, AlgAlts [(con,bs,rhs')] NoDefault)) | scrut_var_lvl `ltMajLvl` lvl -> @@ -296,12 +305,7 @@ floatExpr sw env lvl (Case scrut alts) Nothing -> Level 0 0 Just lvl -> unTopify lvl - END OF CASE FLOATING DROPPED -} - - (_, (fsa, fda, alts')) -> - - (add_stats fse fsa, fda ++ fde, Case scrut' alts') - } + END OF CASE FLOATING DROPPED -} where incd_lvl = incMinorLvl lvl @@ -347,13 +351,13 @@ floatExpr sw env lvl (Case scrut alts) bs' = map fst bs new_env = growIdEnvList env bs in - case (floatExpr sw new_env incd_lvl rhs) of { (fs, rhs_floats, rhs') -> + case (floatExpr new_env incd_lvl rhs) of { (fs, rhs_floats, rhs') -> case (partition_fn incd_lvl rhs_floats) of { (rhs_floats', heres) -> (fs, rhs_floats', (con, bs', install heres rhs')) }} -------------- float_prim_alt (lit, rhs) - = case (floatExpr sw env incd_lvl rhs) of { (fs, rhs_floats, rhs') -> + = case (floatExpr env incd_lvl rhs) of { (fs, rhs_floats, rhs') -> case (partition_fn incd_lvl rhs_floats) of { (rhs_floats', heres) -> (fs, rhs_floats', (lit, install heres rhs')) }} @@ -361,7 +365,7 @@ floatExpr sw env lvl (Case scrut alts) float_deflt NoDefault = (zero_stats, [], NoDefault) float_deflt (BindDefault (b,lvl) rhs) - = case (floatExpr sw new_env lvl rhs) of { (fs, rhs_floats, rhs') -> + = case (floatExpr new_env lvl rhs) of { (fs, rhs_floats, rhs') -> case (partition_fn incd_lvl rhs_floats) of { (rhs_floats', heres) -> (fs, rhs_floats', BindDefault b (install heres rhs')) }} where diff --git a/ghc/compiler/simplCore/FoldrBuildWW.lhs b/ghc/compiler/simplCore/FoldrBuildWW.lhs index a3a8a6ab54..7c97d54151 100644 --- a/ghc/compiler/simplCore/FoldrBuildWW.lhs +++ b/ghc/compiler/simplCore/FoldrBuildWW.lhs @@ -108,7 +108,7 @@ try_split_bind id expr = | FBGoodProd == prod -> {- || any (== FBGoodConsum) consum -} let - (use_args,big_args,args,body) = digForLambdas expr' + (use_args,big_args,args,body) = collectBinders expr' in if length args /= length consum -- funny number of arguments then returnWw [(id,expr')] diff --git a/ghc/compiler/simplCore/LiberateCase.lhs b/ghc/compiler/simplCore/LiberateCase.lhs index 4c17f204d0..2b46c88dc6 100644 --- a/ghc/compiler/simplCore/LiberateCase.lhs +++ b/ghc/compiler/simplCore/LiberateCase.lhs @@ -1,14 +1,21 @@ % -% (c) The AQUA Project, Glasgow University, 1994 +% (c) The AQUA Project, Glasgow University, 1994-1996 % \section[LiberateCase]{Unroll recursion to allow evals to be lifted from a loop} +96/03: We aren't using this at the moment \begin{code} #include "HsVersions.h" module LiberateCase ( liberateCase ) where +import Ubiq{-uitous-} +import Util ( panic ) + +liberateCase = panic "LiberateCase.liberateCase: ToDo" + +{- LATER: to end of file: import CoreUnfold ( UnfoldingGuidance(..) ) import Id ( localiseId, toplevelishId{-debugging-} ) import Maybes @@ -327,4 +334,5 @@ freeScruts (LibCaseEnv bomb lvl lvl_env rec_env scruts) rec_bind_lvl = not (null free_scruts) where free_scruts = [v | (v,lvl) <- scruts, lvl > rec_bind_lvl] +-} \end{code} diff --git a/ghc/compiler/simplCore/SAT.lhs b/ghc/compiler/simplCore/SAT.lhs index dbd4f54000..28cb54cebb 100644 --- a/ghc/compiler/simplCore/SAT.lhs +++ b/ghc/compiler/simplCore/SAT.lhs @@ -1,5 +1,5 @@ % -% (c) The GRASP/AQUA Project, Glasgow University, 1992-1995 +% (c) The GRASP/AQUA Project, Glasgow University, 1992-1996 % %************************************************************************ %* * @@ -7,6 +7,8 @@ %* * %************************************************************************ +96/03: We aren't using the static-argument transformation right now. + May be seen as removing invariants from loops: Arguments of recursive functions that do not change in recursive calls are removed from the recursion, which is done locally @@ -38,11 +40,14 @@ Experimental Evidence: Heap: +/- 7% \begin{code} #include "HsVersions.h" -module SAT ( - doStaticArgs +module SAT ( doStaticArgs ) where + +import Ubiq{-uitous-} +import Util ( panic ) - -- and to make the interface self-sufficient... - ) where +doStaticArgs = panic "SAT.doStaticArgs (ToDo)" + +{- LATER: to end of file: import Maybes ( Maybe(..) ) import SATMonad @@ -205,5 +210,5 @@ getAppArgs app get e = satExpr e `thenSAT` \ e2 -> returnSAT (e2, Nothing) +-} \end{code} - diff --git a/ghc/compiler/simplCore/SATMonad.lhs b/ghc/compiler/simplCore/SATMonad.lhs index 265df4886f..b61deb36a4 100644 --- a/ghc/compiler/simplCore/SATMonad.lhs +++ b/ghc/compiler/simplCore/SATMonad.lhs @@ -1,5 +1,5 @@ % -% (c) The GRASP/AQUA Project, Glasgow University, 1992-1995 +% (c) The GRASP/AQUA Project, Glasgow University, 1992-1996 % %************************************************************************ %* * @@ -7,9 +7,20 @@ %* * %************************************************************************ +96/03: We aren't using the static-argument transformation right now. + \begin{code} #include "HsVersions.h" +module SATMonad where + +import Ubiq{-uitous-} +import Util ( panic ) + +junk_from_SATMonad = panic "SATMonad.junk" + +{- LATER: to end of file: + module SATMonad ( SATInfo(..), updSAEnv, SatM(..), initSAT, emptyEnvSAT, @@ -20,7 +31,7 @@ module SATMonad ( ) where import Type ( mkTyVarTy, mkSigmaTy, TyVarTemplate, - extractTyVarsFromTy, splitSigmaTy, splitTyArgs, + splitSigmaTy, splitTyArgs, glueTyArgs, instantiateTy, TauType(..), Class, ThetaType(..), SigmaType(..), InstTyEnv(..) @@ -135,7 +146,7 @@ newSATName id ty us env getArgLists :: CoreExpr -> ([Arg Type],[Arg Id]) getArgLists expr = let - (uvs, tvs, lambda_bounds, body) = digForLambdas expr + (uvs, tvs, lambda_bounds, body) = collectBinders expr in ([ Static (mkTyVarTy tv) | tv <- tvs ], [ Static v | v <- lambda_bounds ]) @@ -201,7 +212,7 @@ saTransform binder rhs -- this binder *will* get inlined but if it happen to be -- a top level binder it is never removed as dead code, -- therefore we have to remove that information (of it being - -- top-level or exported somehow. + -- top-level or exported somehow.) -- A better fix is to use binder directly but with the TopLevel -- tag (or Exported tag) modified. fake_binder = mkSysLocal @@ -250,4 +261,5 @@ dropStatics (_:args) (t:ts) = t:dropStatics args ts isStatic :: Arg a -> Bool isStatic NotStatic = False isStatic _ = True +-} \end{code} diff --git a/ghc/compiler/simplCore/SetLevels.lhs b/ghc/compiler/simplCore/SetLevels.lhs index 32453a0a25..b52c6035b6 100644 --- a/ghc/compiler/simplCore/SetLevels.lhs +++ b/ghc/compiler/simplCore/SetLevels.lhs @@ -21,23 +21,36 @@ module SetLevels ( -- not exported: , incMajorLvl, isTopMajLvl, unTopify ) where -import Type ( isPrimType, isLeakFreeType, mkTyVarTy, - quantifyTy, TyVarTemplate -- Needed for quantifyTy - ) +import Ubiq{-uitous-} + import AnnCoreSyn -import Literal ( Literal(..) ) -import CmdLineOpts ( GlobalSwitch(..) ) -import FreeVars -import Id ( mkSysLocal, idType, eqId, - isBottomingId, toplevelishId, DataCon(..) - IF_ATTACK_PRAGMAS(COMMA bottomIsGuaranteed) +import CoreSyn + +import CoreUtils ( coreExprType, manifestlyWHNF, manifestlyBottom ) +import FreeVars -- all of it +import Id ( idType, mkSysLocal, toplevelishId, + nullIdEnv, addOneToIdEnv, growIdEnvList, + unionManyIdSets, minusIdSet, mkIdSet, + idSetToList, + lookupIdEnv, IdEnv(..) + ) +import Pretty ( ppStr, ppBesides, ppChar, ppInt ) +import SrcLoc ( mkUnknownSrcLoc ) +import Type ( isPrimType, mkTyVarTys ) +import TyVar ( nullTyVarEnv, addOneToTyVarEnv, + growTyVarEnvList, lookupTyVarEnv, + tyVarSetToList, + TyVarEnv(..), + unionManyTyVarSets ) -import Maybes ( Maybe(..) ) -import Pretty -- debugging only -import UniqSet -import SrcLoc ( mkUnknownSrcLoc, SrcLoc ) -import UniqSupply -import Util +import UniqSupply ( thenUs, returnUs, mapUs, mapAndUnzipUs, + mapAndUnzip3Us, getUnique, UniqSM(..) + ) +import Usage ( UVar(..) ) +import Util ( mapAccumL, zipWithEqual, panic, assertPanic ) + +quantifyTy = panic "SetLevels.quantifyTy (ToDo)" +isLeakFreeType = panic "SetLevels.isLeakFreeType (ToDo)" \end{code} %************************************************************************ @@ -47,19 +60,18 @@ import Util %************************************************************************ \begin{code} -data Level = Level - Int -- Level number of enclosing lambdas - Int -- Number of big-lambda and/or case expressions between - -- here and the nearest enclosing lambda - - | Top -- Means *really* the top level. +data Level + = Top -- Means *really* the top level. + | Level Int -- Level number of enclosing lambdas + Int -- Number of big-lambda and/or case expressions between + -- here and the nearest enclosing lambda \end{code} The {\em level number} on a (type-)lambda-bound variable is the -nesting depth of the (type-)lambda which binds it. On an expression, it's the -maximum level number of its free (type-)variables. On a let(rec)-bound -variable, it's the level of its RHS. On a case-bound variable, it's -the number of enclosing lambdas. +nesting depth of the (type-)lambda which binds it. On an expression, +it's the maximum level number of its free (type-)variables. On a +let(rec)-bound variable, it's the level of its RHS. On a case-bound +variable, it's the number of enclosing lambdas. Top-level variables: level~0. Those bound on the RHS of a top-level definition but ``before'' a lambda; e.g., the \tr{x} in (levels shown @@ -69,24 +81,25 @@ a_0 = let b_? = ... in x_1 = ... b ... in ... \end{verbatim} -Level 0 0 will make something get floated to a top-level "equals", @Top@ -makes it go right to the top. +Level 0 0 will make something get floated to a top-level "equals", +@Top@ makes it go right to the top. -The main function @lvlExpr@ carries a ``context level'' (@ctxt_lvl@). That's -meant to be the level number of the enclosing binder in the final (floated) -program. If the level number of a sub-expression is less than that of the -context, then it might be worth let-binding the sub-expression so that it -will indeed float. This context level starts at @Level 0 0@; it is never @Top@. +The main function @lvlExpr@ carries a ``context level'' (@ctxt_lvl@). +That's meant to be the level number of the enclosing binder in the +final (floated) program. If the level number of a sub-expression is +less than that of the context, then it might be worth let-binding the +sub-expression so that it will indeed float. This context level starts +at @Level 0 0@; it is never @Top@. \begin{code} -type LevelledExpr = GenCoreExpr (Id, Level) Id -type LevelledAtom = GenCoreAtom Id -type LevelledBind = GenCoreBinding (Id, Level) Id +type LevelledExpr = GenCoreExpr (Id, Level) Id TyVar UVar +type LevelledArg = GenCoreArg Id TyVar UVar +type LevelledBind = GenCoreBinding (Id, Level) Id TyVar UVar type LevelEnvs = (IdEnv Level, -- bind Ids to levels TyVarEnv Level) -- bind type variables to levels -tOP_LEVEL = Top +tOP_LEVEL = Top incMajorLvl :: Level -> Level incMajorLvl Top = Level 1 0 @@ -106,11 +119,11 @@ maxLvl l1@(Level maj1 min1) l2@(Level maj2 min2) ltLvl :: Level -> Level -> Bool ltLvl l1 Top = False ltLvl Top (Level _ _) = True -ltLvl (Level maj1 min1) (Level maj2 min2) = (maj1 < maj2) || - (maj1 == maj2 && min1 < min2) +ltLvl (Level maj1 min1) (Level maj2 min2) + = (maj1 < maj2) || (maj1 == maj2 && min1 < min2) -ltMajLvl :: Level -> Level -> Bool -- Tells if one level belongs to a difft - -- *lambda* level to another +ltMajLvl :: Level -> Level -> Bool + -- Tells if one level belongs to a difft *lambda* level to another ltMajLvl l1 Top = False ltMajLvl Top (Level 0 _) = False ltMajLvl Top (Level _ _) = True @@ -120,7 +133,7 @@ isTopLvl :: Level -> Bool isTopLvl Top = True isTopLvl other = False -isTopMajLvl :: Level -> Bool -- Tells if it's the top *lambda* level +isTopMajLvl :: Level -> Bool -- Tells if it's the top *lambda* level isTopMajLvl Top = True isTopMajLvl (Level maj _) = maj == 0 @@ -141,12 +154,11 @@ instance Outputable Level where \begin{code} setLevels :: [CoreBinding] - -> (GlobalSwitch -> Bool) -- access to all global cmd-line opts -> UniqSupply -> [LevelledBind] -setLevels binds sw us - = do_them binds sw us +setLevels binds us + = do_them binds us where -- "do_them"'s main business is to thread the monad along -- It gives each top binding the same empty envt, because @@ -161,25 +173,12 @@ setLevels binds sw us initial_envs = (nullIdEnv, nullTyVarEnv) --- OLDER: lvlTopBind (NonRec binder rhs) - = lvlBind (Level 0 0) initial_envs (AnnCoNonRec binder (freeVars rhs)) + = lvlBind (Level 0 0) initial_envs (AnnNonRec binder (freeVars rhs)) -- Rhs can have no free vars! lvlTopBind (Rec pairs) - = lvlBind (Level 0 0) initial_envs (AnnCoRec [(b,freeVars rhs) | (b,rhs) <- pairs]) - -{- NEWER: Too bad about the types: WDP: -lvlTopBind (NonRec binder rhs) - = {-SIGH:wrong type: ASSERT(isEmptyUniqSet (freeVarsOf rhs))-} -- Rhs can have no free vars! - lvlBind (Level 0 0) initial_envs (AnnCoNonRec binder emptyUniqSet) - -lvlTopBind (Rec pairs) - = lvlBind (Level 0 0) initial_envs - (AnnCoRec [(b, emptyUniqSet) - | (b, rhs) <- pairs, - {-SIGH:ditto:ASSERT(isEmptyUniqSet (freeVarsOf rhs))-} True]) --} + = lvlBind (Level 0 0) initial_envs (AnnRec [(b,freeVars rhs) | (b,rhs) <- pairs]) \end{code} %************************************************************************ @@ -191,14 +190,14 @@ lvlTopBind (Rec pairs) The binding stuff works for top level too. \begin{code} -type CoreBindingWithFVs = AnnCoreBinding Id Id FVInfo +type CoreBindingWithFVs = AnnCoreBinding Id Id TyVar UVar FVInfo lvlBind :: Level -> LevelEnvs -> CoreBindingWithFVs -> LvlM ([LevelledBind], LevelEnvs) -lvlBind ctxt_lvl envs@(venv, tenv) (AnnCoNonRec name rhs) +lvlBind ctxt_lvl envs@(venv, tenv) (AnnNonRec name rhs) = setFloatLevel True {- Already let-bound -} ctxt_lvl envs rhs ty `thenLvl` \ (final_lvl, rhs') -> let @@ -209,7 +208,7 @@ lvlBind ctxt_lvl envs@(venv, tenv) (AnnCoNonRec name rhs) ty = idType name -lvlBind ctxt_lvl envs@(venv, tenv) (AnnCoRec pairs) +lvlBind ctxt_lvl envs@(venv, tenv) (AnnRec pairs) = decideRecFloatLevel ctxt_lvl envs binders rhss `thenLvl` \ (final_lvl, extra_binds, rhss') -> let @@ -252,43 +251,42 @@ don't want @lvlExpr@ to turn the scrutinee of the @case@ into an MFE If there were another lambda in @r@'s rhs, it would get level-2 as well. \begin{code} -lvlExpr _ _ (_, AnnCoVar v) = returnLvl (Var v) -lvlExpr _ _ (_, AnnCoLit l) = returnLvl (Lit l) -lvlExpr _ _ (_, AnnCoCon con tys atoms) = returnLvl (Con con tys atoms) -lvlExpr _ _ (_, AnnCoPrim op tys atoms) = returnLvl (Prim op tys atoms) +lvlExpr _ _ (_, AnnVar v) = returnLvl (Var v) +lvlExpr _ _ (_, AnnLit l) = returnLvl (Lit l) +lvlExpr _ _ (_, AnnCon con args) = returnLvl (Con con args) +lvlExpr _ _ (_, AnnPrim op args) = returnLvl (Prim op args) -lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoTyApp expr ty) - = lvlExpr ctxt_lvl envs expr `thenLvl` \ expr' -> - returnLvl (CoTyApp expr' ty) - -lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoApp fun arg) +lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnApp fun arg) = lvlExpr ctxt_lvl envs fun `thenLvl` \ fun' -> returnLvl (App fun' arg) -lvlExpr ctxt_lvl envs (_, AnnCoSCC cc expr) +lvlExpr ctxt_lvl envs (_, AnnSCC cc expr) = lvlExpr ctxt_lvl envs expr `thenLvl` \ expr' -> returnLvl (SCC cc expr') -lvlExpr ctxt_lvl (venv, tenv) (_, AnnCoTyLam tyvar e) - = lvlExpr incd_lvl (venv, new_tenv) e `thenLvl` \ e' -> - returnLvl (CoTyLam tyvar e') - where - incd_lvl = incMinorLvl ctxt_lvl - new_tenv = addOneToTyVarEnv tenv tyvar incd_lvl - -lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoLam arg rhs) +lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnLam (ValBinder arg) rhs) = lvlMFE incd_lvl (new_venv, tenv) rhs `thenLvl` \ rhs' -> - returnLvl (Lam (arg,incd_lvl) rhs') + returnLvl (Lam (ValBinder (arg,incd_lvl)) rhs') where incd_lvl = incMajorLvl ctxt_lvl new_venv = growIdEnvList venv [(arg,incd_lvl)] -lvlExpr ctxt_lvl envs (_, AnnCoLet bind body) +lvlExpr ctxt_lvl (venv, tenv) (_, AnnLam (TyBinder tyvar) e) + = lvlExpr incd_lvl (venv, new_tenv) e `thenLvl` \ e' -> + returnLvl (Lam (TyBinder tyvar) e') + where + incd_lvl = incMinorLvl ctxt_lvl + new_tenv = addOneToTyVarEnv tenv tyvar incd_lvl + +lvlExpr ctxt_lvl (venv, tenv) (_, AnnLam (UsageBinder u) e) + = panic "SetLevels.lvlExpr:AnnLam UsageBinder" + +lvlExpr ctxt_lvl envs (_, AnnLet bind body) = lvlBind ctxt_lvl envs bind `thenLvl` \ (binds', new_envs) -> lvlExpr ctxt_lvl new_envs body `thenLvl` \ body' -> returnLvl (foldr Let body' binds') -- mkCoLet* requires Core... -lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoCase expr alts) +lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCase expr alts) = lvlMFE ctxt_lvl envs expr `thenLvl` \ expr' -> lvl_alts alts `thenLvl` \ alts' -> returnLvl (Case expr' alts') @@ -296,7 +294,7 @@ lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoCase expr alts) expr_type = coreExprType (deAnnotate expr) incd_lvl = incMinorLvl ctxt_lvl - lvl_alts (AnnCoAlgAlts alts deflt) + lvl_alts (AnnAlgAlts alts deflt) = mapLvl lvl_alt alts `thenLvl` \ alts' -> lvl_deflt deflt `thenLvl` \ deflt' -> returnLvl (AlgAlts alts' deflt') @@ -309,7 +307,7 @@ lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoCase expr alts) lvlMFE incd_lvl new_envs e `thenLvl` \ e' -> returnLvl (con, bs', e') - lvl_alts (AnnCoPrimAlts alts deflt) + lvl_alts (AnnPrimAlts alts deflt) = mapLvl lvl_alt alts `thenLvl` \ alts' -> lvl_deflt deflt `thenLvl` \ deflt' -> returnLvl (PrimAlts alts' deflt') @@ -318,9 +316,9 @@ lvlExpr ctxt_lvl envs@(venv, tenv) (_, AnnCoCase expr alts) = lvlMFE incd_lvl envs e `thenLvl` \ e' -> returnLvl (lit, e') - lvl_deflt AnnCoNoDefault = returnLvl NoDefault + lvl_deflt AnnNoDefault = returnLvl NoDefault - lvl_deflt (AnnCoBindDefault b expr) + lvl_deflt (AnnBindDefault b expr) = let new_envs = (addOneToIdEnv venv b incd_lvl, tenv) in @@ -436,8 +434,8 @@ setFloatLevel alreadyLetBound ctxt_lvl envs@(venv, tenv) -- The truth: better to give it expr_lvl in case it is pinning -- something non-trivial which depends on it. where - fv_list = uniqSetToList fvs - tv_list = uniqSetToList tfvs + fv_list = idSetToList fvs + tv_list = tyVarSetToList tfvs expr_lvl = ids_only_lvl `maxLvl` tyvars_only_lvl ids_only_lvl = foldr (maxLvl . idLevel venv) tOP_LEVEL fv_list tyvars_only_lvl = foldr (maxLvl . tyvarLevel tenv) tOP_LEVEL tv_list @@ -453,9 +451,10 @@ setFloatLevel alreadyLetBound ctxt_lvl envs@(venv, tenv) de_ann_expr = deAnnotate expr - is_trivial (CoTyApp e _) = is_trivial e - is_trivial (Var _) = True - is_trivial _ = False + is_trivial (App e a) + | notValArg a = is_trivial e + is_trivial (Var _) = True + is_trivial _ = False offending_tyvars = filter offending tv_list --non_offending_tyvars = filter (not . offending) tv_list @@ -508,9 +507,9 @@ abstractWrtTyVars offending_tyvars ty (venv,tenv) lvl expr = lvlExpr incd_lvl new_envs expr `thenLvl` \ expr' -> newLvlVar poly_ty `thenLvl` \ poly_var -> let - poly_var_rhs = mkCoTyLam offending_tyvars expr' + poly_var_rhs = mkTyLam offending_tyvars expr' poly_var_binding = NonRec (poly_var, lvl) poly_var_rhs - poly_var_app = mkCoTyApps (Var poly_var) (map mkTyVarTy offending_tyvars) + poly_var_app = mkTyApp (Var poly_var) (mkTyVarTys offending_tyvars) final_expr = Let poly_var_binding poly_var_app -- mkCoLet* requires Core in returnLvl final_expr @@ -607,12 +606,12 @@ decideRecFloatLevel ctxt_lvl envs@(venv, tenv) ids rhss -- The "d_rhss" are the right-hand sides of "D" and "D'" -- in the documentation above - d_rhss = [ mkCoTyApps (Var poly_var) offending_tyvar_tys | poly_var <- poly_vars] + d_rhss = [ mkTyApp (Var poly_var) offending_tyvar_tys | poly_var <- poly_vars] -- "local_binds" are "D'" in the documentation above local_binds = zipWithEqual NonRec ids_w_incd_lvl d_rhss - poly_var_rhss = [ mkCoTyLam offending_tyvars (foldr Let rhs' local_binds) + poly_var_rhss = [ mkTyLam offending_tyvars (foldr Let rhs' local_binds) | rhs' <- rhss' -- mkCoLet* requires Core... ] @@ -635,10 +634,10 @@ decideRecFloatLevel ctxt_lvl envs@(venv, tenv) ids rhss where tys = map idType ids - fvs = unionManyUniqSets [freeVarsOf rhs | rhs <- rhss] `minusUniqSet` mkUniqSet ids - tfvs = unionManyUniqSets [freeTyVarsOf rhs | rhs <- rhss] - fv_list = uniqSetToList fvs - tv_list = uniqSetToList tfvs + fvs = unionManyIdSets [freeVarsOf rhs | rhs <- rhss] `minusIdSet` mkIdSet ids + tfvs = unionManyTyVarSets [freeTyVarsOf rhs | rhs <- rhss] + fv_list = idSetToList fvs + tv_list = tyVarSetToList tfvs ids_only_lvl = foldr (maxLvl . idLevel venv) tOP_LEVEL fv_list tyvars_only_lvl = foldr (maxLvl . tyvarLevel tenv) tOP_LEVEL tv_list @@ -648,7 +647,7 @@ decideRecFloatLevel ctxt_lvl envs@(venv, tenv) ids rhss | ids_only_lvl `ltLvl` tyvars_only_lvl = filter offending tv_list | otherwise = [] - offending_tyvar_tys = map mkTyVarTy offending_tyvars + offending_tyvar_tys = mkTyVarTys offending_tyvars poly_tys = [ snd (quantifyTy offending_tyvars ty) | ty <- tys ] @@ -675,11 +674,14 @@ isWorthFloating alreadyLetBound expr ********** -} isWorthFloatingExpr :: CoreExpr -> Bool -isWorthFloatingExpr (Var v) = False -isWorthFloatingExpr (Lit lit) = False -isWorthFloatingExpr (Con con tys []) = False -- Just a type application -isWorthFloatingExpr (CoTyApp expr ty) = isWorthFloatingExpr expr -isWorthFloatingExpr other = True + +isWorthFloatingExpr (Var v) = False +isWorthFloatingExpr (Lit lit) = False +isWorthFloatingExpr (App e arg) + | notValArg arg = isWorthFloatingExpr e +isWorthFloatingExpr (Con con as) + | all notValArg as = False -- Just a type application +isWorthFloatingExpr _ = True canFloatToTop :: (Type, CoreExprWithFVs) -> Bool @@ -719,33 +721,13 @@ tyvarLevel tenv tyvar %************************************************************************ \begin{code} -type LvlM result - = (GlobalSwitch -> Bool) -> UniqSupply -> result - -thenLvl m k sw us - = case splitUniqSupply us of { (s1, s2) -> - case m sw s1 of { m_result -> - k m_result sw s2 }} - -returnLvl v sw us = v - -mapLvl f [] = returnLvl [] -mapLvl f (x:xs) - = f x `thenLvl` \ r -> - mapLvl f xs `thenLvl` \ rs -> - returnLvl (r:rs) - -mapAndUnzipLvl f [] = returnLvl ([], []) -mapAndUnzipLvl f (x:xs) - = f x `thenLvl` \ (r1, r2) -> - mapAndUnzipLvl f xs `thenLvl` \ (rs1, rs2) -> - returnLvl (r1:rs1, r2:rs2) - -mapAndUnzip3Lvl f [] = returnLvl ([], [], []) -mapAndUnzip3Lvl f (x:xs) - = f x `thenLvl` \ (r1, r2, r3) -> - mapAndUnzip3Lvl f xs `thenLvl` \ (rs1, rs2, rs3) -> - returnLvl (r1:rs1, r2:rs2, r3:rs3) +type LvlM result = UniqSM result + +thenLvl = thenUs +returnLvl = returnUs +mapLvl = mapUs +mapAndUnzipLvl = mapAndUnzipUs +mapAndUnzip3Lvl = mapAndUnzip3Us \end{code} We create a let-binding for `interesting' (non-utterly-trivial) @@ -754,9 +736,6 @@ applications, to give them a fighting chance of being floated. \begin{code} newLvlVar :: Type -> LvlM Id -newLvlVar ty sw us - = id - where - id = mkSysLocal SLIT("lvl") uniq ty mkUnknownSrcLoc - uniq = getUnique us +newLvlVar ty us + = mkSysLocal SLIT("lvl") (getUnique us) ty mkUnknownSrcLoc \end{code} diff --git a/ghc/compiler/simplCore/SimplCase.lhs b/ghc/compiler/simplCore/SimplCase.lhs index d2cb6c5e61..7c70bca19d 100644 --- a/ghc/compiler/simplCore/SimplCase.lhs +++ b/ghc/compiler/simplCore/SimplCase.lhs @@ -20,7 +20,7 @@ import PrelInfo ( getPrimOpResultInfo, PrimOpResultInfo(..), PrimOp, ) import Type ( splitSigmaTy, splitTyArgs, glueTyArgs, getTyConFamilySize, isPrimType, - maybeDataTyCon + maybeAppDataTyCon ) import Literal ( isNoRepLit, Literal ) import CmdLineOpts ( SimplifierSwitch(..) ) @@ -463,7 +463,7 @@ bindLargeRhs env args rhs_ty rhs_c let final_rhs = (if switchIsSet new_env SimplDoEtaReduction - then mkCoLamTryingEta + then mkValLamTryingEta else mkValLam) used_args' rhs' in returnSmpl (NonRec rhs_fun_id final_rhs, @@ -789,7 +789,7 @@ mkCoCase scrut (AlgAlts outer_alts v | scrut_is_var = Var scrut_var | otherwise = Con con arg_tys (map VarArg args) - arg_tys = case maybeDataTyCon (idType deflt_var) of + arg_tys = case maybeAppDataTyCon (idType deflt_var) of Just (_, arg_tys, _) -> arg_tys mkCoCase scrut (PrimAlts diff --git a/ghc/compiler/simplCore/SimplCore.lhs b/ghc/compiler/simplCore/SimplCore.lhs index cf446c0564..2ada37315e 100644 --- a/ghc/compiler/simplCore/SimplCore.lhs +++ b/ghc/compiler/simplCore/SimplCore.lhs @@ -473,12 +473,12 @@ calcInlinings scc_s_OK sw_chkr inline_env_so_far top_binds rhs_looks_like_a_caf = not (manifestlyWHNF rhs) rhs_looks_like_a_data_val - = case (digForLambdas rhs) of + = case (collectBinders rhs) of (_, _, [], Con _ _ _) -> True other -> False rhs_arg_tys - = case (digForLambdas rhs) of + = case (collectBinders rhs) of (_, _, val_binders, _) -> map idType val_binders (mentioned_ids, _, _, mentions_litlit) diff --git a/ghc/compiler/simplCore/SimplUtils.lhs b/ghc/compiler/simplCore/SimplUtils.lhs index d1bd744fce..3f5c1a5f2f 100644 --- a/ghc/compiler/simplCore/SimplUtils.lhs +++ b/ghc/compiler/simplCore/SimplUtils.lhs @@ -10,7 +10,7 @@ module SimplUtils ( floatExposesHNF, - mkCoTyLamTryingEta, mkCoLamTryingEta, + mkTyLamTryingEta, mkValLamTryingEta, etaExpandCount, @@ -21,33 +21,21 @@ module SimplUtils ( type_ok_for_let_to_case ) where -IMPORT_Trace -- ToDo: rm (debugging) -import Pretty +import Ubiq{-uitous-} +import BinderInfo +import CoreSyn +import CoreUtils ( manifestlyWHNF ) +import Id ( idType, isBottomingId, getIdArity ) +import IdInfo ( arityMaybe ) +import Maybes ( maybeToBool ) +import PrelInfo ( augmentId, buildId, realWorldStateTy ) import SimplEnv import SimplMonad +import Type ( isPrimType, maybeAppDataTyCon, getTyVar_maybe ) +import Util ( isIn, panic ) -import BinderInfo - -import PrelInfo ( primOpIsCheap, realWorldStateTy, - buildId, augmentId - IF_ATTACK_PRAGMAS(COMMA realWorldTy) - IF_ATTACK_PRAGMAS(COMMA tagOf_PrimOp) - IF_ATTACK_PRAGMAS(COMMA pprPrimOp) - ) -import Type ( extractTyVarsFromTy, getTyVarMaybe, isPrimType, - splitTypeWithDictsAsArgs, maybeDataTyCon, - applyTy, isFunType, TyVar, TyVarTemplate - ) -import Id ( getInstantiatedDataConSig, isDataCon, idType, - getIdArity, isBottomingId, idWantsToBeINLINEd, - DataCon(..), Id - ) -import IdInfo -import CmdLineOpts ( SimplifierSwitch(..) ) -import Maybes ( maybeToBool, Maybe(..) ) -import Outputable -- isExported ... -import Util +primOpIsCheap = panic "SimplUtils. (ToDo)" \end{code} @@ -79,8 +67,8 @@ floatExposesHNF float_lets float_primops ok_to_dup rhs -- because it *will* become one. -- likewise for `augment g h' -- - try (App (CoTyApp (Var bld) _) _) | bld == buildId = True - try (App (App (CoTyApp (Var bld) _) _) _) | bld == augmentId = True + try (App (App (Var bld) _) _) | bld == buildId = True + try (App (App (App (Var aug) _) _) _) | aug == augmentId = True try other = manifestlyWHNF other {- but *not* necessarily "manifestlyBottom other"... @@ -99,7 +87,7 @@ floatExposesHNF float_lets float_primops ok_to_dup rhs to allocate it eagerly as that's a waste. -} - try_alt (lit,rhs) = try rhs + try_alt (lit,rhs) = try rhs try_deflt NoDefault = False try_deflt (BindDefault _ rhs) = try rhs @@ -127,13 +115,13 @@ gives rise to a recursive function for the list comprehension, and f turns out to be just a single call to this recursive function. \begin{code} -mkCoLamTryingEta :: [Id] -- Args to the lambda +mkValLamTryingEta :: [Id] -- Args to the lambda -> CoreExpr -- Lambda body -> CoreExpr -mkCoLamTryingEta [] body = body +mkValLamTryingEta [] body = body -mkCoLamTryingEta orig_ids body +mkValLamTryingEta orig_ids body = reduce_it (reverse orig_ids) body where bale_out = mkValLam orig_ids body @@ -150,16 +138,18 @@ mkCoLamTryingEta orig_ids body reduce_it ids other = bale_out - is_elem = isIn "mkCoLamTryingEta" + is_elem = isIn "mkValLamTryingEta" ----------- residual_ok :: CoreExpr -> Bool -- Checks for type application - -- and function not one of the - -- bound vars - residual_ok (CoTyApp fun ty) = residual_ok fun - residual_ok (Var v) = not (v `is_elem` orig_ids) -- Fun mustn't be one of - -- the bound ids - residual_ok other = False + -- and function not one of the + -- bound vars + + residual_ok (Var v) = not (v `is_elem` orig_ids) + -- Fun mustn't be one of the bound ids + residual_ok (App fun arg) + | notValArg arg = residual_ok fun + residual_ok other = False \end{code} Eta expansion @@ -169,20 +159,22 @@ such that E ===> (\x1::t1 x1::t2 ... xn::tn -> E x1 x2 ... xn) -is a safe transformation. In particular, the transformation should not -cause work to be duplicated, unless it is ``cheap'' (see @manifestlyCheap@ below). +is a safe transformation. In particular, the transformation should +not cause work to be duplicated, unless it is ``cheap'' (see +@manifestlyCheap@ below). -@etaExpandCount@ errs on the conservative side. It is always safe to return 0. +@etaExpandCount@ errs on the conservative side. It is always safe to +return 0. An application of @error@ is special, because it can absorb as many -arguments as you care to give it. For this special case we return 100, -to represent "infinity", which is a bit of a hack. +arguments as you care to give it. For this special case we return +100, to represent "infinity", which is a bit of a hack. \begin{code} etaExpandCount :: GenCoreExpr bdr Id - -> Int -- Number of extra args you can safely abstract + -> Int -- Number of extra args you can safely abstract -etaExpandCount (Lam _ body) +etaExpandCount (Lam (ValBinder _) body) = 1 + etaExpandCount body etaExpandCount (Let bind body) @@ -193,37 +185,38 @@ etaExpandCount (Case scrut alts) | manifestlyCheap scrut = minimum [etaExpandCount rhs | rhs <- rhssOfAlts alts] -etaExpandCount (App fun _) = case etaExpandCount fun of - 0 -> 0 - n -> n-1 -- Knock off one - -etaExpandCount fun@(CoTyApp _ _) = eta_fun fun etaExpandCount fun@(Var _) = eta_fun fun +etaExpandCount (App fun arg) + | notValArg arg = eta_fun fun + | otherwise = case etaExpandCount fun of + 0 -> 0 + n -> n-1 -- Knock off one -etaExpandCount other = 0 -- Give up +etaExpandCount other = 0 -- Give up -- Lit, Con, Prim, - -- CoTyLam, + -- non-val Lam, -- Scc (pessimistic; ToDo), -- Let with non-whnf rhs(s), -- Case with non-whnf scrutinee +----------------------------- eta_fun :: GenCoreExpr bdr Id -- The function -> Int -- How many args it can safely be applied to -eta_fun (CoTyApp fun ty) = eta_fun fun +eta_fun (App fun arg) | notValArg arg = eta_fun fun eta_fun expr@(Var v) - | isBottomingId v -- Bottoming ids have "infinite arity" - = 10000 -- Blargh. Infinite enough! + | isBottomingId v -- Bottoming ids have "infinite arity" + = 10000 -- Blargh. Infinite enough! eta_fun expr@(Var v) - | maybeToBool arity_maybe -- We know the arity + | maybeToBool arity_maybe -- We know the arity = arity where arity_maybe = arityMaybe (getIdArity v) arity = case arity_maybe of { Just arity -> arity } -eta_fun other = 0 -- Give up +eta_fun other = 0 -- Give up \end{code} @manifestlyCheap@ looks at a Core expression and returns \tr{True} if @@ -252,10 +245,11 @@ manifestlyCheap :: GenCoreExpr bndr Id -> Bool manifestlyCheap (Var _) = True manifestlyCheap (Lit _) = True manifestlyCheap (Con _ _ _) = True -manifestlyCheap (Lam _ _) = True -manifestlyCheap (CoTyLam _ e) = manifestlyCheap e manifestlyCheap (SCC _ e) = manifestlyCheap e +manifestlyCheap (Lam (ValBinder _) _) = True +manifestlyCheap (Lam other_binder e) = manifestlyCheap e + manifestlyCheap (Prim op _ _) = primOpIsCheap op manifestlyCheap (Let bind body) @@ -268,20 +262,20 @@ manifestlyCheap other_expr -- look for manifest partial application = case (collectArgs other_expr) of { (fun, args) -> case fun of - Var f | isBottomingId f -> True -- Application of a function which - -- always gives bottom; we treat this as - -- a WHNF, because it certainly doesn't - -- need to be shared! + Var f | isBottomingId f -> True -- Application of a function which + -- always gives bottom; we treat this as + -- a WHNF, because it certainly doesn't + -- need to be shared! Var f -> let - num_val_args = length [ a | (ValArg a) <- args ] - in - num_val_args == 0 || -- Just a type application of - -- a variable (f t1 t2 t3) - -- counts as WHNF - case (arityMaybe (getIdArity f)) of - Nothing -> False - Just arity -> num_val_args < arity + num_val_args = numValArgs args + in + num_val_args == 0 || -- Just a type application of + -- a variable (f t1 t2 t3) + -- counts as WHNF + case (arityMaybe (getIdArity f)) of + Nothing -> False + Just arity -> num_val_args < arity _ -> False } @@ -321,9 +315,9 @@ applications since this breaks the specialiser: /\ a -> f Char# a =NO=> f Char# \begin{code} -mkCoTyLamTryingEta :: [TyVar] -> CoreExpr -> CoreExpr +mkTyLamTryingEta :: [TyVar] -> CoreExpr -> CoreExpr -mkCoTyLamTryingEta tyvars tylam_body +mkTyLamTryingEta tyvars tylam_body = if tyvars == tyvar_args && -- Same args in same order check_fun fun -- Function left is ok @@ -332,15 +326,18 @@ mkCoTyLamTryingEta tyvars tylam_body fun else -- The vastly common case - mkCoTyLam tyvars tylam_body + mkTyLam tyvars tylam_body where (tyvar_args, fun) = strip_tyvar_args [] tylam_body - strip_tyvar_args args_so_far tyapp@(CoTyApp fun ty) - = case getTyVarMaybe ty of + strip_tyvar_args args_so_far tyapp@(App fun (TyArg ty)) + = case getTyVar_maybe ty of Just tyvar_arg -> strip_tyvar_args (tyvar_arg:args_so_far) fun Nothing -> (args_so_far, tyapp) + strip_tyvar_args args_so_far (App _ (UsageArg _)) + = panic "SimplUtils.mkTyLamTryingEta: strip_tyvar_args UsageArg" + strip_tyvar_args args_so_far fun = (args_so_far, fun) @@ -373,7 +370,7 @@ mkIdentityAlts rhs_ty returnSmpl (PrimAlts [] (BindDefault (binder, bad_occ_info) (Var binder))) | otherwise - = case maybeDataTyCon rhs_ty of + = case (maybeAppDataTyCon rhs_ty) of Just (tycon, ty_args, [data_con]) -> -- algebraic type suitable for unpacking let (_,inst_con_arg_tys,_) = getInstantiatedDataConSig data_con ty_args @@ -406,7 +403,7 @@ simplIdWantsToBeINLINEd id env type_ok_for_let_to_case :: Type -> Bool type_ok_for_let_to_case ty - = case maybeDataTyCon ty of + = case (maybeAppDataTyCon ty) of Nothing -> False Just (tycon, ty_args, []) -> False Just (tycon, ty_args, non_null_data_cons) -> True diff --git a/ghc/compiler/simplCore/SimplVar.lhs b/ghc/compiler/simplCore/SimplVar.lhs index 10a9f3caa0..c0a91cddea 100644 --- a/ghc/compiler/simplCore/SimplVar.lhs +++ b/ghc/compiler/simplCore/SimplVar.lhs @@ -199,7 +199,7 @@ considerUnfolding env var args txt_occ form_summary template guidance rhs_looks_like_a_Con = let - (_,_,val_binders,body) = digForLambdas template + (_,_,val_binders,body) = collectBinders template in case (val_binders, body) of ([], Con _ _ _) -> True diff --git a/ghc/compiler/simplCore/Simplify.lhs b/ghc/compiler/simplCore/Simplify.lhs index fe5f6aebfd..36591fc7de 100644 --- a/ghc/compiler/simplCore/Simplify.lhs +++ b/ghc/compiler/simplCore/Simplify.lhs @@ -21,7 +21,7 @@ import PrelInfo ( getPrimOpResultInfo, PrimOpResultInfo(..), IF_ATTACK_PRAGMAS(COMMA tagOf_PrimOp) IF_ATTACK_PRAGMAS(COMMA pprPrimOp) ) -import Type ( maybeDataTyCon, mkTyVarTy, applyTy, +import Type ( maybeAppDataTyCon, mkTyVarTy, mkTyVarTys, applyTy, splitTyArgs, splitTypeWithDictsAsArgs, maybeUnpackFunTy, isPrimType ) @@ -349,7 +349,7 @@ Type lambdas We only eta-reduce a type lambda if all type arguments in the body can be eta-reduced. This requires us to collect up all tyvar parameters so -we can pass them all to @mkCoTyLamTryingEta@. +we can pass them all to @mkTyLamTryingEta@. \begin{code} simplExpr env (CoTyLam tyvar body) (TypeArg ty : args) @@ -375,7 +375,7 @@ simplExpr env tylam@(CoTyLam tyvar body) [] = simplExpr env body [] `thenSmpl` \ body' -> returnSmpl ( (if switchIsSet env SimplDoEtaReduction - then mkCoTyLamTryingEta + then mkTyLamTryingEta else mkCoTyLam) (reverse tyvars') body' ) @@ -548,7 +548,7 @@ simplRhsExpr env binder@(id,occ_info) rhs = -- Deal with the big lambda part mapSmpl cloneTyVarSmpl tyvars `thenSmpl` \ tyvars' -> let - lam_env = extendTyEnvList rhs_env (tyvars `zip` (map mkTyVarTy tyvars')) + lam_env = extendTyEnvList rhs_env (tyvars `zip` (mkTyVarTys tyvars')) in -- Deal with the little lambda part -- Note that we call simplLam even if there are no binders, in case @@ -558,7 +558,7 @@ simplRhsExpr env binder@(id,occ_info) rhs -- Put it back together returnSmpl ( (if switchIsSet env SimplDoEtaReduction - then mkCoTyLamTryingEta + then mkTyLamTryingEta else mkCoTyLam) tyvars' lambda' ) where @@ -569,7 +569,7 @@ simplRhsExpr env binder@(id,occ_info) rhs rhs_env | simplIdWantsToBeINLINEd id env = filterUnfoldEnvForInlines env | otherwise = env - (uvars, tyvars, binders, body) = digForLambdas rhs + (uvars, tyvars, binders, body) = collectBinders rhs min_no_of_args | not (null binders) && -- It's not a thunk switchIsSet env SimplDoArityExpand -- Arity expansion on @@ -618,7 +618,7 @@ simplLam env binders body min_no_of_args simplExpr new_env body [] `thenSmpl` \ body' -> returnSmpl ( (if switchIsSet new_env SimplDoEtaReduction - then mkCoLamTryingEta + then mkValLamTryingEta else mkValLam) binders' body' ) @@ -632,7 +632,7 @@ simplLam env binders body min_no_of_args simplExpr new_env body (map (ValArg.VarArg) extra_binders') `thenSmpl` \ body' -> returnSmpl ( (if switchIsSet new_env SimplDoEtaReduction - then mkCoLamTryingEta + then mkValLamTryingEta else mkValLam) (binders' ++ extra_binders') body' ) diff --git a/ghc/compiler/simplStg/StgSATMonad.lhs b/ghc/compiler/simplStg/StgSATMonad.lhs index 1da8207597..5996c18cb8 100644 --- a/ghc/compiler/simplStg/StgSATMonad.lhs +++ b/ghc/compiler/simplStg/StgSATMonad.lhs @@ -14,8 +14,8 @@ module StgSATMonad ( getArgLists, saTransform ) where -import Type ( mkTyVarTy, mkSigmaTy, TyVarTemplate, - extractTyVarsFromTy, splitSigmaTy, splitTyArgs, +import Type ( mkSigmaTy, TyVarTemplate, + splitSigmaTy, splitTyArgs, glueTyArgs, instantiateTy, TauType(..), Class, ThetaType(..), SigmaType(..), InstTyEnv(..) diff --git a/ghc/compiler/specialise/Specialise.lhs b/ghc/compiler/specialise/Specialise.lhs index e503a9c373..e96941a549 100644 --- a/ghc/compiler/specialise/Specialise.lhs +++ b/ghc/compiler/specialise/Specialise.lhs @@ -965,7 +965,7 @@ addDictBinds dbinders dbind (UsageDetails db_cis db_tycon_cis db_dbs db_fvs db_c -- to look at the type of the dictionary itself. -- Doing the proper job would entail keeping track of free tyvars as -- well as free vars, which would be a bore. - db_ftvs = mkUniqSet (extractTyVarsFromTys (map idType dbinders)) + db_ftvs = tyVarsOfTypes (map idType dbinders) \end{code} %************************************************************************ diff --git a/ghc/compiler/stgSyn/StgLint.lhs b/ghc/compiler/stgSyn/StgLint.lhs index 29faa874ce..b97ef11d10 100644 --- a/ghc/compiler/stgSyn/StgLint.lhs +++ b/ghc/compiler/stgSyn/StgLint.lhs @@ -172,7 +172,7 @@ lintStgExpr e@(StgCase scrut _ _ _ alts) = lintStgExpr scrut `thenMaybeL` \ _ -> -- Check that it is a data type - case maybeDataTyCon scrut_ty of + case maybeAppDataTyCon scrut_ty of Nothing -> addErrL (mkCaseDataConMsg e) `thenL_` returnL Nothing Just (tycon, _, _) @@ -218,7 +218,7 @@ lintStgAlts alts scrut_ty case_tycon Just _ -> returnL () -- that's cool lintAlgAlt scrut_ty (con, args, _, rhs) - = (case maybeDataTyCon scrut_ty of + = (case maybeAppDataTyCon scrut_ty of Nothing -> addErrL (mkAlgAltMsg1 scrut_ty) Just (tycon, tys_applied, cons) -> diff --git a/ghc/compiler/stranal/SaAbsInt.lhs b/ghc/compiler/stranal/SaAbsInt.lhs index affcbfb142..156f2ae1c1 100644 --- a/ghc/compiler/stranal/SaAbsInt.lhs +++ b/ghc/compiler/stranal/SaAbsInt.lhs @@ -25,7 +25,7 @@ import PrelInfo ( PrimOp(..), floatTyCon, wordTyCon, addrTyCon, PrimRep ) -import Type ( isPrimType, maybeDataTyCon, +import Type ( isPrimType, maybeAppDataTyCon, maybeSingleConstructorTyCon, returnsRealWorld, isEnumerationTyCon, TyVarTemplate, TyCon @@ -833,7 +833,7 @@ findRecDemand strflags seen str_fn abs_fn ty else -- It's strict (or we're pretending it is)! - case maybeDataTyCon ty of + case maybeAppDataTyCon ty of Nothing -> wwStrict @@ -874,7 +874,7 @@ findRecDemand strflags seen str_fn abs_fn ty (all_strict, num_strict) = strflags is_numeric_type ty - = case (maybeDataTyCon ty) of -- NB: duplicates stuff done above + = case (maybeAppDataTyCon ty) of -- NB: duplicates stuff done above Nothing -> False Just (tycon, _, _) | tycon `is_elem` diff --git a/ghc/compiler/stranal/StrictAnal.lhs b/ghc/compiler/stranal/StrictAnal.lhs index f98e5e4285..6605d26262 100644 --- a/ghc/compiler/stranal/StrictAnal.lhs +++ b/ghc/compiler/stranal/StrictAnal.lhs @@ -400,7 +400,7 @@ addStrictnessInfoToId strflags str_val abs_val binder body if (isBot str_val) then binder `addIdStrictness` mkBottomStrictnessInfo else - case (digForLambdas body) of { (_, _, lambda_bounds, rhs) -> + case (collectBinders body) of { (_, _, lambda_bounds, rhs) -> let tys = map idType lambda_bounds strictness = findStrictness strflags tys str_val abs_val diff --git a/ghc/compiler/stranal/WorkWrap.lhs b/ghc/compiler/stranal/WorkWrap.lhs index bda7de10b1..a82579db1d 100644 --- a/ghc/compiler/stranal/WorkWrap.lhs +++ b/ghc/compiler/stranal/WorkWrap.lhs @@ -206,7 +206,7 @@ tryWW fn_id rhs -- OK, it looks as if a worker is worth a try let - (uvars, tyvars, args, body) = digForLambdas rhs + (uvars, tyvars, args, body) = collectBinders rhs body_ty = coreExprType body in uniqSMtoWwM (mkWwBodies body_ty tyvars args args_info) `thenWw` \ result -> diff --git a/ghc/compiler/stranal/WwLib.lhs b/ghc/compiler/stranal/WwLib.lhs index b87bd4c61c..4fa859a4e4 100644 --- a/ghc/compiler/stranal/WwLib.lhs +++ b/ghc/compiler/stranal/WwLib.lhs @@ -31,8 +31,8 @@ import IdInfo -- lots of things import Maybes ( maybeToBool, Maybe(..), MaybeErr ) import SaLib import SrcLoc ( mkUnknownSrcLoc ) -import Type ( mkTyVarTy, mkFunTys, isPrimType, - maybeDataTyCon, quantifyTy +import Type ( mkTyVarTys, mkFunTys, isPrimType, + maybeAppDataTyCon, quantifyTy ) import UniqSupply -} @@ -230,7 +230,7 @@ mkWwBodies body_ty tyvars args arg_infos wrapper_w_hole = \ worker_id -> mkLam tyvars args ( wrap_frag ( - mkCoTyApps (Var worker_id) (map mkTyVarTy tyvars) + mkCoTyApps (Var worker_id) (mkTyVarTys tyvars) )) worker_w_hole = \ orig_body -> @@ -326,7 +326,7 @@ mk_ww_arg_processing (arg : args) (WwUnpack cmpnt_infos : infos) max_extra_args | new_max_extra_args > 0 -- Check that we are prepared to add arguments = -- this is the complicated one. --pprTrace "Unpack; num_wrkr_args=" (ppCat [ppInt num_wrkr_args, ppStr "; new_max=", ppInt new_num_wrkr_args, ppStr "; arg=", ppr PprDebug arg, ppr PprDebug (WwUnpack cmpnt_infos)]) ( - case maybeDataTyCon arg_ty of + case maybeAppDataTyCon arg_ty of Nothing -> -- Not a data type panic "mk_ww_arg_processing: not datatype" diff --git a/ghc/compiler/typecheck/GenSpecEtc.lhs b/ghc/compiler/typecheck/GenSpecEtc.lhs index f0008df8d0..27e4a00550 100644 --- a/ghc/compiler/typecheck/GenSpecEtc.lhs +++ b/ghc/compiler/typecheck/GenSpecEtc.lhs @@ -18,14 +18,14 @@ import Ubiq import TcMonad import Inst ( Inst, InstOrigin(..), LIE(..), plusLIE, newDicts, tyVarsOfInst, instToId ) -import TcEnv ( tcGetGlobalTyVars, newMonoIds ) +import TcEnv ( tcGetGlobalTyVars ) import TcSimplify ( tcSimplify, tcSimplifyAndCheck, tcSimplifyWithExtraGlobals ) import TcType ( TcType(..), TcThetaType(..), TcTauType(..), TcTyVarSet(..), TcTyVar(..), tcInstType, zonkTcType ) import HsSyn ( HsBinds(..), Bind(..), MonoBinds(..), HsExpr, OutPat(..), - Sig, HsLit, ArithSeqInfo, InPat, GRHSsAndBinds, Match, Fake, - collectBinders ) + Sig, HsLit, ArithSeqInfo, InPat, GRHSsAndBinds, Match, Fake + ) import TcHsSyn ( TcIdOcc(..), TcIdBndr(..), TcHsBinds(..), TcBind(..), TcExpr(..) ) import Bag ( Bag, foldBag, bagToList, listToBag, isEmptyBag ) diff --git a/ghc/compiler/typecheck/TcBinds.lhs b/ghc/compiler/typecheck/TcBinds.lhs index a61b07552e..9ecbe7f330 100644 --- a/ghc/compiler/typecheck/TcBinds.lhs +++ b/ghc/compiler/typecheck/TcBinds.lhs @@ -39,7 +39,8 @@ import Maybes ( assocMaybe, catMaybes, Maybe(..) ) import Outputable ( pprNonOp ) import PragmaInfo ( PragmaInfo(..) ) import Pretty -import Type ( mkTyVarTy, isTyVarTy, mkSigmaTy, splitSigmaTy, +import Type ( mkTyVarTy, mkTyVarTys, isTyVarTy, + mkSigmaTy, splitSigmaTy, splitRhoTy, mkForAllTy, splitForAllTy ) import Util ( panic ) \end{code} @@ -401,7 +402,7 @@ tcPragmaSig (SpecSig name poly_ty maybe_spec_name src_loc) let (main_tyvars, main_rho) = splitForAllTy main_ty (main_theta,main_tau) = splitRhoTy main_rho - main_arg_tys = map mkTyVarTy main_tyvars + main_arg_tys = mkTyVarTys main_tyvars in -- Check that the specialised type is indeed an instance of diff --git a/ghc/compiler/typecheck/TcClassDcl.lhs b/ghc/compiler/typecheck/TcClassDcl.lhs index 805fe986c8..7bb5dc7678 100644 --- a/ghc/compiler/typecheck/TcClassDcl.lhs +++ b/ghc/compiler/typecheck/TcClassDcl.lhs @@ -47,7 +47,7 @@ import Pretty import PprType ( GenType, GenTyVar, GenClassOp ) import SpecEnv ( SpecEnv(..) ) import SrcLoc ( mkGeneratedSrcLoc ) -import Type ( mkFunTy, mkTyVarTy, mkDictTy, +import Type ( mkFunTy, mkTyVarTy, mkTyVarTys, mkDictTy, mkForAllTy, mkSigmaTy, splitSigmaTy) import TysWiredIn ( stringTy ) import TyVar ( GenTyVar ) @@ -283,7 +283,7 @@ buildSelectors :: Class -- The class object buildSelectors clas clas_tyvar scs sc_sel_ids ops op_sel_ids = -- Make new Ids for the components of the dictionary - mapNF_Tc (tcInstType [] . getClassOpLocalType) ops `thenNF_Tc` \ op_tys -> + mapNF_Tc (tcInstType [] . getClassOpLocalType) ops `thenNF_Tc` \ op_tys -> newLocalIds (map getClassOpString ops) op_tys `thenNF_Tc` \ method_ids -> @@ -296,13 +296,11 @@ buildSelectors clas clas_tyvar scs sc_sel_ids ops op_sel_ids -- Make suitable bindings for the selectors let - tc_method_ids = map TcId method_ids - mk_sel sel_id method_or_dict - = mkSelBind sel_id clas_tyvar clas_dict dict_ids tc_method_ids method_or_dict + = mkSelBind sel_id clas_tyvar clas_dict dict_ids method_ids method_or_dict in - listNF_Tc (zipWithEqual mk_sel op_sel_ids tc_method_ids) `thenNF_Tc` \ op_sel_binds -> - listNF_Tc (zipWithEqual mk_sel sc_sel_ids dict_ids) `thenNF_Tc` \ sc_sel_binds -> + listNF_Tc (zipWithEqual mk_sel op_sel_ids method_ids) `thenNF_Tc` \ op_sel_binds -> + listNF_Tc (zipWithEqual mk_sel sc_sel_ids dict_ids) `thenNF_Tc` \ sc_sel_binds -> returnNF_Tc (SingleBind ( NonRecBind ( @@ -366,7 +364,7 @@ mkSelBind :: Id -- the selector id mkSelBind sel_id clas_tyvar clas_dict dicts methods method_or_dict@(TcId op) = let (op_tyvars,op_theta,op_tau) = splitSigmaTy (idType op) - op_tys = map mkTyVarTy op_tyvars + op_tys = mkTyVarTys op_tyvars in newDicts ClassDeclOrigin op_theta `thenNF_Tc` \ (_, op_dicts) -> diff --git a/ghc/compiler/typecheck/TcDeriv.lhs b/ghc/compiler/typecheck/TcDeriv.lhs index 253bb98bc2..8912626138 100644 --- a/ghc/compiler/typecheck/TcDeriv.lhs +++ b/ghc/compiler/typecheck/TcDeriv.lhs @@ -24,6 +24,7 @@ import TcHsSyn ( TcIdOcc ) import TcMonad import Inst ( InstOrigin(..), InstanceMapper(..) ) import TcEnv ( getEnv_TyCons ) +import TcKind ( TcKind ) import TcGenDeriv -- Deriv stuff import TcInstUtil ( InstInfo(..), mkInstanceRelatedIds, buildInstanceEnvs ) import TcSimplify ( tcSimplifyThetas ) @@ -47,7 +48,7 @@ import ProtoName ( eqProtoName, ProtoName(..), Name ) import SrcLoc ( mkGeneratedSrcLoc, mkUnknownSrcLoc, SrcLoc ) import TyCon ( getTyConTyVars, getTyConDataCons, getTyConDerivings, maybeTyConSingleCon, isEnumerationTyCon, TyCon ) -import Type ( GenType(..), TauType(..), mkTyVarTy, applyTyCon, +import Type ( GenType(..), TauType(..), mkTyVarTys, applyTyCon, mkSigmaTy, mkDictTy, isPrimType, instantiateTy, getAppTyCon, getAppDataTyCon ) import TyVar ( GenTyVar ) @@ -249,7 +250,7 @@ makeDerivEqns :: TcM s [DerivEqn] makeDerivEqns = tcGetEnv `thenNF_Tc` \ env -> let - tycons = eltsUFM (getEnv_TyCons env) + tycons = getEnv_TyCons env think_about_deriving = need_deriving tycons in mapTc (chk_out think_about_deriving) think_about_deriving `thenTc_` @@ -303,7 +304,7 @@ makeDerivEqns = (clas, tycon, tyvars, constraints) where tyvars = getTyConTyVars tycon -- ToDo: Do we need new tyvars ??? - tyvar_tys = map mkTyVarTy tyvars + tyvar_tys = mkTyVarTys tyvars data_cons = getTyConDataCons tycon constraints = concat (map mk_constraints data_cons) @@ -420,7 +421,7 @@ add_solns modname inst_infos_in eqns solns all_inst_infos = inst_infos_in `unionBags` listToBag new_inst_infos mk_deriv_inst_info (clas, tycon, tyvars, _) theta - = InstInfo clas tyvars (applyTyCon tycon (map mkTyVarTy tyvars)) + = InstInfo clas tyvars (applyTyCon tycon (mkTyVarTys tyvars)) theta theta -- Blarg. This is the dfun_theta slot, -- which is needed by buildInstanceEnv; diff --git a/ghc/compiler/typecheck/TcEnv.lhs b/ghc/compiler/typecheck/TcEnv.lhs index c2b831dcaa..42a6c9b3f9 100644 --- a/ghc/compiler/typecheck/TcEnv.lhs +++ b/ghc/compiler/typecheck/TcEnv.lhs @@ -6,14 +6,16 @@ module TcEnv( initEnv, getEnv_LocalIds, getEnv_TyCons, getEnv_Classes, - tcExtendKindEnv, tcExtendTyVarEnv, tcExtendTyConEnv, tcExtendClassEnv, - tcLookupTyVar, tcLookupTyCon, tcLookupClass, tcLookupClassByKey, + tcTyVarScope, tcTyVarScopeGivenKinds, tcLookupTyVar, + + tcExtendTyConEnv, tcLookupTyCon, tcLookupTyConByKey, + tcExtendClassEnv, tcLookupClass, tcLookupClassByKey, tcExtendGlobalValEnv, tcExtendLocalValEnv, - tcLookupLocalValue, tcLookupLocalValueOK, + tcLookupLocalValue, tcLookupLocalValueOK, tcLookupLocalValueByKey, tcLookupGlobalValue, tcLookupGlobalValueByKey, - tcTyVarScope, newMonoIds, newLocalIds, + newMonoIds, newLocalIds, newLocalId, tcGetGlobalTyVars ) where @@ -22,12 +24,12 @@ import Ubiq import TcMLoop -- for paranoia checking import Id ( Id(..), GenId, idType, mkUserLocal ) -import TcHsSyn ( TcIdBndr(..) ) +import TcHsSyn ( TcIdBndr(..), TcIdOcc(..) ) import TcKind ( TcKind, newKindVars, tcKindToKind, kindToTcKind ) import TcType ( TcType(..), TcMaybe, TcTyVar(..), TcTyVarSet(..), newTyVarTys, zonkTcTyVars ) import TyVar ( mkTyVar, getTyVarKind, unionTyVarSets, emptyTyVarSet ) import Type ( tyVarsOfTypes ) -import TyCon ( TyCon, getTyConKind ) +import TyCon ( TyCon, Arity(..), getTyConKind, getSynTyConArity ) import Class ( Class(..), GenClass, getClassSig ) import TcMonad @@ -46,135 +48,126 @@ Data type declarations \begin{code} data TcEnv s = TcEnv (TyVarEnv s) + (TyConEnv s) + (ClassEnv s) (ValueEnv Id) -- Globals (ValueEnv (TcIdBndr s)) -- Locals (MutableVar s (TcTyVarSet s)) -- Free type variables of locals -- ...why mutable? see notes with tcGetGlobalTyVars - (KindEnv s) -- Gives TcKinds of TyCons and Classes - TyConEnv - ClassEnv type TyVarEnv s = UniqFM (TcKind s, TyVar) -type TyConEnv = UniqFM TyCon -type KindEnv s = UniqFM (TcKind s) -type ClassEnv = UniqFM Class +type TyConEnv s = UniqFM (TcKind s, Maybe Arity, TyCon) -- Arity present for Synonyms only +type ClassEnv s = UniqFM (TcKind s, Class) type ValueEnv id = UniqFM id initEnv :: MutableVar s (TcTyVarSet s) -> TcEnv s -initEnv mut = TcEnv emptyUFM emptyUFM emptyUFM mut emptyUFM emptyUFM emptyUFM +initEnv mut = TcEnv emptyUFM emptyUFM emptyUFM emptyUFM emptyUFM mut -getEnv_LocalIds (TcEnv _ _ ls _ _ _ _) = ls -getEnv_TyCons (TcEnv _ _ _ _ _ ts _) = ts -getEnv_Classes (TcEnv _ _ _ _ _ _ cs) = cs +getEnv_LocalIds (TcEnv _ _ _ _ ls _) = eltsUFM ls +getEnv_TyCons (TcEnv _ ts _ _ _ _) = [tycon | (_, _, tycon) <- eltsUFM ts] +getEnv_Classes (TcEnv _ _ cs _ _ _) = [clas | (_, clas) <- eltsUFM cs] \end{code} Making new TcTyVars, with knot tying! ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ \begin{code} -tcTyVarScope :: [Name] -- Names of some type variables - -> ([TyVar] -> TcM s a) -- Thing to type check in their scope - -> TcM s a -- Result - -tcTyVarScope tyvar_names thing_inside - = newKindVars (length tyvar_names) `thenNF_Tc` \ tyvar_kinds -> +tcTyVarScopeGivenKinds + :: [Name] -- Names of some type variables + -> [TcKind s] + -> ([TyVar] -> TcM s a) -- Thing to type check in their scope + -> TcM s a -- Result - fixTc (\ ~(tyvars, _) -> - -- Ok to look at kinds, but not tyvars! - tcExtendTyVarEnv tyvar_names (tyvar_kinds `zipLazy` tyvars) ( +tcTyVarScopeGivenKinds names kinds thing_inside + = fixTc (\ ~(rec_tyvars, _) -> + -- Ok to look at names, kinds, but not tyvars! - -- Do the thing inside - thing_inside tyvars `thenTc` \ result -> + tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> + let + tve' = addListToUFM tve (names `zip` (kinds `zipLazy` rec_tyvars)) + in + tcSetEnv (TcEnv tve' tce ce gve lve gtvs) + (thing_inside rec_tyvars) `thenTc` \ result -> -- Get the tyvar's Kinds from their TcKinds - mapNF_Tc tcKindToKind tyvar_kinds `thenNF_Tc` \ tyvar_kinds' -> + mapNF_Tc tcKindToKind kinds `thenNF_Tc` \ kinds' -> -- Construct the real TyVars let - tyvars = zipWithEqual mk_tyvar tyvar_names tyvar_kinds' + tyvars = zipWithEqual mk_tyvar names kinds' mk_tyvar name kind = mkTyVar (getNameShortName name) (getItsUnique name) kind in returnTc (tyvars, result) - )) `thenTc` \ (_,result) -> + ) `thenTc` \ (_,result) -> returnTc result + +tcTyVarScope names thing_inside + = newKindVars (length names) `thenNF_Tc` \ kinds -> + tcTyVarScopeGivenKinds names kinds thing_inside \end{code} The Kind, TyVar, Class and TyCon envs ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -Extending the environments +Extending the environments. Notice the uses of @zipLazy@, which makes sure +that the knot-tied TyVars, TyCons and Classes aren't looked at too early. \begin{code} -tcExtendKindEnv :: [Name] -> [TcKind s] -> TcM s r -> TcM s r -tcExtendKindEnv names kinds scope - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> - let - ke' = addListToUFM ke (names `zip` kinds) - in - tcSetEnv (TcEnv tve gve lve gtvs ke' tce ce) scope - -tcExtendTyVarEnv :: [Name] -> [(TcKind s, TyVar)] -> TcM s r -> TcM s r -tcExtendTyVarEnv tyvar_names kinds_w_tyvars scope - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> - let - tve' = addListToUFM tve (tyvar_names `zip` kinds_w_tyvars) - in - tcSetEnv (TcEnv tve' gve lve gtvs ke tce ce) scope - -tcExtendTyConEnv tycons scope - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> +tcExtendTyConEnv :: [(Name,Maybe Arity)] -> [TyCon] -> TcM s r -> TcM s r +tcExtendTyConEnv names_w_arities tycons scope + = newKindVars (length names_w_arities) `thenNF_Tc` \ kinds -> + tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> let - tce' = addListToUFM_Directly tce [(getItsUnique tycon, tycon) | tycon <- tycons] + tce' = addListToUFM tce [ (name, (kind, arity, tycon)) + | ((name,arity), (kind,tycon)) <- names_w_arities `zip` + (kinds `zipLazy` tycons) + ] in - tcSetEnv (TcEnv tve gve lve gtvs ke tce' ce) scope + tcSetEnv (TcEnv tve tce' ce gve lve gtvs) scope -tcExtendClassEnv classes scope - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> +tcExtendClassEnv :: [Name] -> [Class] -> TcM s r -> TcM s r +tcExtendClassEnv names classes scope + = newKindVars (length names) `thenNF_Tc` \ kinds -> + tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> let - ce' = addListToUFM_Directly ce [(getItsUnique clas, clas) | clas <- classes] + ce' = addListToUFM ce (names `zip` (kinds `zipLazy` classes)) in - tcSetEnv (TcEnv tve gve lve gtvs ke tce ce') scope + tcSetEnv (TcEnv tve tce ce' gve lve gtvs) scope \end{code} -Looking up in the environments +Looking up in the environments. \begin{code} tcLookupTyVar name - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> returnNF_Tc (lookupWithDefaultUFM tve (panic "tcLookupTyVar") name) tcLookupTyCon (WiredInTyCon tc) -- wired in tycons - = returnNF_Tc (kindToTcKind (getTyConKind tc), tc) + = returnNF_Tc (kindToTcKind (getTyConKind tc), getSynTyConArity tc, tc) tcLookupTyCon name - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> - let - tycon = lookupWithDefaultUFM tce (panic "tcLookupTyCon") name - kind = lookupWithDefaultUFM ke (kindToTcKind (getTyConKind tycon)) name - -- The KE will bind tycon in the current mutually-recursive set. - -- If the KE doesn't, then the tycon is already defined, and we - -- can safely grab the kind from the TyCon itself - in - returnNF_Tc (kind,tycon) + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> + returnNF_Tc (lookupWithDefaultUFM tce (panic "tcLookupTyCon") name) +tcLookupTyConByKey uniq + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> + let + (kind, arity, tycon) = lookupWithDefaultUFM_Directly tce (panic "tcLookupTyCon") uniq + in + returnNF_Tc tycon tcLookupClass name - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> - let - clas = lookupWithDefaultUFM ce (panic "tcLookupClass") name - (tyvar, _, _) = getClassSig clas - kind = lookupWithDefaultUFM ke (kindToTcKind (getTyVarKind tyvar)) name - in - returnNF_Tc (kind,clas) + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> + returnNF_Tc (lookupWithDefaultUFM ce (panic "tcLookupClass") name) tcLookupClassByKey uniq - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> let - clas = lookupWithDefaultUFM_Directly ce (panic "tcLookupClas") uniq + (kind, clas) = lookupWithDefaultUFM_Directly ce (panic "tcLookupClas") uniq in - returnNF_Tc (clas) + returnNF_Tc clas \end{code} @@ -183,14 +176,14 @@ Extending and consulting the value environment ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ \begin{code} tcExtendGlobalValEnv ids scope - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> let gve' = addListToUFM_Directly gve [(getItsUnique id, id) | id <- ids] in - tcSetEnv (TcEnv tve gve' lve gtvs ke tce ce) scope + tcSetEnv (TcEnv tve tce ce gve' lve gtvs) scope tcExtendLocalValEnv names ids scope - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> tcReadMutVar gtvs `thenNF_Tc` \ global_tvs -> let lve' = addListToUFM lve (names `zip` ids) @@ -199,7 +192,7 @@ tcExtendLocalValEnv names ids scope in tcNewMutVar new_global_tyvars `thenNF_Tc` \ gtvs' -> - tcSetEnv (TcEnv tve gve lve' gtvs' ke tce ce) scope + tcSetEnv (TcEnv tve tce ce gve lve' gtvs') scope \end{code} @tcGetGlobalTyVars@ returns a fully-zonked set of tyvars free in the environment. @@ -209,7 +202,7 @@ the environment. \begin{code} tcGetGlobalTyVars :: NF_TcM s (TcTyVarSet s) tcGetGlobalTyVars - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> tcReadMutVar gtvs `thenNF_Tc` \ global_tvs -> zonkTcTyVars global_tvs `thenNF_Tc` \ global_tvs' -> tcWriteMutVar gtvs global_tvs' `thenNF_Tc_` @@ -219,12 +212,17 @@ tcGetGlobalTyVars \begin{code} tcLookupLocalValue :: Name -> NF_TcM s (Maybe (TcIdBndr s)) tcLookupLocalValue name - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> returnNF_Tc (lookupUFM lve name) +tcLookupLocalValueByKey :: Unique -> NF_TcM s (Maybe (TcIdBndr s)) +tcLookupLocalValueByKey uniq + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> + returnNF_Tc (lookupUFM_Directly lve uniq) + tcLookupLocalValueOK :: String -> Name -> NF_TcM s (TcIdBndr s) tcLookupLocalValueOK err name - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> returnNF_Tc (lookupWithDefaultUFM lve (panic err) name) @@ -234,7 +232,7 @@ tcLookupGlobalValue (WiredInVal id) -- wired in ids = returnNF_Tc id tcLookupGlobalValue name - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> returnNF_Tc (lookupWithDefaultUFM gve def name) where #ifdef DEBUG @@ -246,7 +244,7 @@ tcLookupGlobalValue name tcLookupGlobalValueByKey :: Unique -> NF_TcM s Id tcLookupGlobalValueByKey uniq - = tcGetEnv `thenNF_Tc` \ (TcEnv tve gve lve gtvs ke tce ce) -> + = tcGetEnv `thenNF_Tc` \ (TcEnv tve tce ce gve lve gtvs) -> returnNF_Tc (lookupWithDefaultUFM_Directly gve def uniq) where #ifdef DEBUG @@ -275,13 +273,19 @@ newMonoIds names kind m where no_of_names = length names -newLocalIds :: [FAST_STRING] -> [TcType s] -> NF_TcM s [TcIdBndr s] +newLocalId :: FAST_STRING -> TcType s -> NF_TcM s (TcIdOcc s) +newLocalId name ty + = tcGetSrcLoc `thenNF_Tc` \ loc -> + tcGetUnique `thenNF_Tc` \ uniq -> + returnNF_Tc (TcId (mkUserLocal name uniq ty loc)) + +newLocalIds :: [FAST_STRING] -> [TcType s] -> NF_TcM s [TcIdOcc s] newLocalIds names tys = tcGetSrcLoc `thenNF_Tc` \ loc -> tcGetUniques (length names) `thenNF_Tc` \ uniqs -> let new_ids = zipWith3Equal mk_id names uniqs tys - mk_id name uniq ty = mkUserLocal name uniq ty loc + mk_id name uniq ty = TcId (mkUserLocal name uniq ty loc) in returnNF_Tc new_ids \end{code} diff --git a/ghc/compiler/typecheck/TcExpr.lhs b/ghc/compiler/typecheck/TcExpr.lhs index f6fc5be286..9f911d4b00 100644 --- a/ghc/compiler/typecheck/TcExpr.lhs +++ b/ghc/compiler/typecheck/TcExpr.lhs @@ -32,6 +32,7 @@ import TcSimplify ( tcSimplifyAndCheck, tcSimplifyRank2 ) import TcType ( TcType(..), TcMaybe(..), tcReadTyVar, tcInstType, tcInstTcType, tcInstTyVar, newTyVarTy, zonkTcTyVars ) +import TcKind ( TcKind ) import Class ( Class(..), getClassSig ) import Id ( Id(..), GenId, idType ) @@ -41,11 +42,11 @@ import PrelInfo ( intPrimTy, charPrimTy, doublePrimTy, floatPrimTy, addrPrimTy, addrTy, boolTy, charTy, stringTy, mkListTy, mkTupleTy, mkPrimIoTy ) -import Type ( mkFunTy, mkAppTy, mkTyVarTy, +import Type ( mkFunTy, mkAppTy, mkTyVarTy, mkTyVarTys, getTyVar_maybe, getFunTy_maybe, splitForAllTy, splitRhoTy, splitSigmaTy, isTauTy, mkFunTys, tyVarsOfType, getForAllTy_maybe ) -import TyVar ( GenTyVar, TyVarSet(..), unionTyVarSets, tyVarListToSet ) +import TyVar ( GenTyVar, TyVarSet(..), unionTyVarSets, mkTyVarSet ) import Unify ( unifyTauTy, unifyTauTyList, unifyTauTyLists ) import Unique ( Unique, cCallableClassKey, cReturnableClassKey, enumFromClassOpKey, enumFromThenClassOpKey, @@ -432,7 +433,7 @@ tcExpr in_expr@(ExprWithTySig expr poly_ty) -- Check overloading constraints tcSimplifyAndCheck - (tyVarListToSet sig_tyvars') + (mkTyVarSet sig_tyvars') sig_dicts lie `thenTc_` -- If everything is ok, return the stuff unchanged, except for @@ -576,7 +577,7 @@ tcArg expected_arg_ty arg -- Even if there isn't, there may be some Insts which mention the arg_tyvars, -- but which, on simplification, don't actually need a dictionary involving -- the tyvar. So we have to do a proper simplification right here. - tcSimplifyRank2 (tyVarListToSet arg_tyvars') + tcSimplifyRank2 (mkTyVarSet arg_tyvars') lie_arg `thenTc` \ (free_insts, inst_binds) -> -- This HsLet binds any Insts which came out of the simplification. @@ -616,7 +617,7 @@ tcId name let (tyvars, rho) = splitForAllTy ty (theta,tau) = splitRhoTy rho - arg_tys = map mkTyVarTy tyvars + arg_tys = mkTyVarTys tyvars in -- Is it overloaded? case theta of diff --git a/ghc/compiler/typecheck/TcInstDcls.lhs b/ghc/compiler/typecheck/TcInstDcls.lhs index 2f75b9d0c4..6e3db5bc9d 100644 --- a/ghc/compiler/typecheck/TcInstDcls.lhs +++ b/ghc/compiler/typecheck/TcInstDcls.lhs @@ -36,7 +36,7 @@ import Inst ( Inst, InstOrigin(..), InstanceMapper(..), newDicts, newMethod, LIE(..), emptyLIE, plusLIE ) import TcBinds ( tcPragmaSigs ) import TcDeriv ( tcDeriving ) -import TcEnv ( tcLookupClass, tcTyVarScope, newLocalIds ) +import TcEnv ( tcLookupClass, tcTyVarScope, newLocalId ) import TcGRHSs ( tcGRHSsAndBinds ) import TcInstUtil ( InstInfo(..), mkInstanceRelatedIds, buildInstanceEnvs ) import TcKind ( TcKind, unifyKind ) @@ -57,7 +57,7 @@ import Class ( GenClass, GenClassOp, isCcallishClass, getClassBigSig, getClassOps, getClassOpLocalType ) import CoreUtils ( escErrorMsg ) -import Id ( idType, isDefaultMethodId_maybe ) +import Id ( GenId, idType, isDefaultMethodId_maybe ) import ListSetOps ( minusList ) import Maybes ( maybeToBool, expectJust ) import Name ( Name, getTagFromClassOpName ) @@ -69,10 +69,10 @@ import PprStyle import Pretty import RnUtils ( GlobalNameMappers(..), GlobalNameMapper(..) ) import TyCon ( derivedFor ) -import Type ( GenType(..), ThetaType(..), mkTyVarTy, - splitSigmaTy, splitAppTy, isTyVarTy, matchTy, +import Type ( GenType(..), ThetaType(..), mkTyVarTys, + splitSigmaTy, splitAppTy, isTyVarTy, matchTy, mkSigmaTy, getTyCon_maybe, maybeBoxedPrimType ) -import TyVar ( GenTyVar, tyVarListToSet ) +import TyVar ( GenTyVar, mkTyVarSet ) import TysWiredIn ( stringTy ) import Unique ( Unique ) import Util ( panic ) @@ -348,7 +348,7 @@ tcInstDecl2 (InstInfo clas inst_tyvars inst_ty -- Get the class signature mapNF_Tc tcInstTyVar inst_tyvars `thenNF_Tc` \ inst_tyvars' -> let - tenv = inst_tyvars `zip` (map mkTyVarTy inst_tyvars') + tenv = inst_tyvars `zip` (mkTyVarTys inst_tyvars') (class_tyvar, super_classes, sc_sel_ids, @@ -360,7 +360,7 @@ tcInstDecl2 (InstInfo clas inst_tyvars inst_ty let sc_theta' = super_classes `zip` (repeat inst_ty') origin = InstanceDeclOrigin - mk_method sel_id = newMethod origin (RealId sel_id) [inst_ty'] + mk_method sel_id = newMethodId sel_id inst_ty' origin locn in -- Create dictionary Ids from the specified instance contexts. newDicts origin sc_theta' `thenNF_Tc` \ (sc_dicts, sc_dict_ids) -> @@ -392,7 +392,7 @@ tcInstDecl2 (InstInfo clas inst_tyvars inst_ty dict_and_method_binds = dict_bind `AndMonoBinds` method_mbinds - inst_tyvars_set' = tyVarListToSet inst_tyvars' + inst_tyvars_set' = mkTyVarSet inst_tyvars' in -- Check the overloading constraints of the methods and superclasses tcAddErrCtxt (bindSigCtxt meth_ids) ( @@ -439,7 +439,55 @@ tcInstDecl2 (InstInfo clas inst_tyvars inst_ty returnTc (const_lie `plusLIE` spec_lie, inst_binds) \end{code} -This function makes a default method which calls the global default method, at +@mkMethodId@ manufactures an id for a local method. +It's rather turgid stuff, because there are two cases: + + (a) For methods with no local polymorphism, we can make an Inst of the + class-op selector function and a corresp InstId; + which is good because then other methods which call + this one will do so directly. + + (b) For methods with local polymorphism, we can't do this. For example, + + class Foo a where + op :: (Num b) => a -> b -> a + + Here the type of the class-op-selector is + + forall a b. (Foo a, Num b) => a -> b -> a + + The locally defined method at (say) type Float will have type + + forall b. (Num b) => Float -> b -> Float + + and the one is not an instance of the other. + + So for these we just make a local (non-Inst) id with a suitable type. + +How disgusting. + +\begin{code} +newMethodId sel_id inst_ty origin loc + = let (sel_tyvars,sel_theta,sel_tau) = splitSigmaTy (idType sel_id) + (_:meth_theta) = sel_theta -- The local theta is all except the + -- first element of the context + in + case sel_tyvars of + -- Ah! a selector for a class op with no local polymorphism + -- Build an Inst for this + [clas_tyvar] -> newMethod origin (RealId sel_id) [inst_ty] + + -- Ho! a selector for a class op with local polymorphism. + -- Just make a suitably typed local id for this + (clas_tyvar:local_tyvars) -> + tcInstType [(clas_tyvar,inst_ty)] + (mkSigmaTy local_tyvars meth_theta sel_tau) + `thenNF_Tc` \ method_ty -> + newLocalId (getOccurrenceName sel_id) method_ty `thenNF_Tc` \ meth_id -> + returnNF_Tc (emptyLIE, meth_id) +\end{code} + +The next function makes a default method which calls the global default method, at the appropriate instance type. See the notes under default decls in TcClassDcl.lhs. @@ -465,7 +513,7 @@ makeInstanceDeclDefaultMethodExpr origin this_dict class_ops defm_ids inst_ty ta mkHsTyLam op_tyvars ( mkHsDictLam op_dicts ( mkHsDictApp (mkHsTyApp (HsVar (RealId defm_id)) - (inst_ty : map mkTyVarTy op_tyvars)) + (inst_ty : mkTyVarTys op_tyvars)) (this_dict : op_dicts) ))) where @@ -640,9 +688,9 @@ processInstBinds1 inst_tyvars avail_insts method_ids mbind -- The latter is needed just so we can return an AbsBinds wrapped -- up inside a MonoBinds. - newLocalIds [occ,occ] [method_tau,method_ty] `thenNF_Tc` \ new_ids -> + newLocalId occ method_tau `thenNF_Tc` \ local_id -> + newLocalId occ method_ty `thenNF_Tc` \ copy_id -> let - [local_id, copy_id] = map TcId new_ids inst_method_tyvars = inst_tyvars ++ method_tyvars in -- Typecheck the method @@ -665,7 +713,7 @@ processInstBinds1 inst_tyvars avail_insts method_ids mbind -- the Bar-ish things. tcAddErrCtxt (methodSigCtxt op method_ty) ( tcSimplifyAndCheck - (tyVarListToSet inst_method_tyvars) + (mkTyVarSet inst_method_tyvars) (method_dicts `plusLIE` avail_insts) lieIop ) `thenTc` \ (f_dicts, dict_binds) -> @@ -747,7 +795,7 @@ tcSpecInstSig e ce tce inst_infos inst_mapper (SpecInstSig class_name ty src_loc babyTcMtoTcM (tcInstanceType ce tce tmpl_e True src_loc ty) `thenTc` \ inst_ty -> let - maybe_tycon = case maybeDataTyCon inst_ty of + maybe_tycon = case maybeAppDataTyCon inst_ty of Just (tc,_,_) -> Just tc Nothing -> Nothing @@ -818,7 +866,7 @@ lookup_unspec_inst clas maybe_tycon inst_infos Just tycon -> match_tycon tycon Nothing -> match_fun - match_tycon tycon inst_ty = case (maybeDataTyCon inst_ty) of + match_tycon tycon inst_ty = case (maybeAppDataTyCon inst_ty) of Just (inst_tc,_,_) -> tycon == inst_tc Nothing -> False @@ -826,7 +874,7 @@ lookup_unspec_inst clas maybe_tycon inst_infos is_plain_instance inst_ty - = case (maybeDataTyCon inst_ty) of + = case (maybeAppDataTyCon inst_ty) of Just (_,tys,_) -> all isTyVarTemplateTy tys Nothing -> case maybeUnpackFunTy inst_ty of Just (arg, res) -> isTyVarTemplateTy arg && isTyVarTemplateTy res diff --git a/ghc/compiler/typecheck/TcInstUtil.lhs b/ghc/compiler/typecheck/TcInstUtil.lhs index 4e6b72dc65..6853735afd 100644 --- a/ghc/compiler/typecheck/TcInstUtil.lhs +++ b/ghc/compiler/typecheck/TcInstUtil.lhs @@ -34,7 +34,7 @@ import PprType ( GenClass, GenType, GenTyVar ) import Pretty import SpecEnv ( SpecEnv(..), nullSpecEnv, addOneToSpecEnv ) import SrcLoc ( SrcLoc ) -import Type ( mkSigmaTy, mkForAllTys, mkDictTy, mkTyVarTy, +import Type ( mkSigmaTy, mkForAllTys, mkDictTy, mkTyVarTys, splitForAllTy, instantiateTy, matchTy, ThetaType(..) ) import TyVar ( GenTyVar ) import Unique ( Unique ) @@ -272,9 +272,9 @@ addClassInstance Succeeded spec_env' -> spec_env' ) where (local_tyvars, _) = splitForAllTy (getClassOpLocalType op) - local_tyvar_tys = map mkTyVarTy local_tyvars + local_tyvar_tys = mkTyVarTys local_tyvars rhs = mkValLam [dict] (mkTyApp (mkTyApp (Var meth_id) - (map mkTyVarTy inst_tyvars)) + (mkTyVarTys inst_tyvars)) local_tyvar_tys) in returnTc (class_inst_env', op_spec_envs') diff --git a/ghc/compiler/typecheck/TcModule.lhs b/ghc/compiler/typecheck/TcModule.lhs index 46668beb82..4daf3b4aa3 100644 --- a/ghc/compiler/typecheck/TcModule.lhs +++ b/ghc/compiler/typecheck/TcModule.lhs @@ -26,7 +26,8 @@ import TcBinds ( tcBindsAndThen ) import TcClassDcl ( tcClassDecls2 ) import TcDefaults ( tcDefaults ) import TcEnv ( tcExtendGlobalValEnv, getEnv_LocalIds, - getEnv_TyCons, getEnv_Classes) + getEnv_TyCons, getEnv_Classes, + tcLookupLocalValueByKey, tcLookupTyConByKey ) import TcIfaceSig ( tcInterfaceSigs ) import TcInstDcls ( tcInstDecls1, tcInstDecls2 ) import TcInstUtil ( buildInstanceEnvs, InstInfo ) @@ -68,10 +69,10 @@ tcModule :: GlobalNameMappers -- final renamer info for derivings [(Id, TypecheckedHsExpr)]), -- constant instance binds - ([RenamedFixityDecl], [Id], UniqFM TyCon, UniqFM Class, Bag InstInfo), + ([RenamedFixityDecl], [Id], [TyCon], [Class], Bag InstInfo), -- things for the interface generator - (UniqFM TyCon, UniqFM Class), + ([TyCon], [Class]), -- environments of info from this module only FiniteMap TyCon [(Bool, [Maybe Type])], @@ -169,10 +170,10 @@ tcModule renamer_name_funs tycons = getEnv_TyCons final_env classes = getEnv_Classes final_env - local_tycons = filterUFM isLocallyDefined tycons - local_classes = filterUFM isLocallyDefined classes + local_tycons = filter isLocallyDefined tycons + local_classes = filter isLocallyDefined classes - exported_ids = [v | v <- eltsUFM localids, + exported_ids = [v | v <- localids, isExported v && not (isDataCon v) && not (isMethodSelId v)] in -- Backsubstitution. Monomorphic top-level decls may have @@ -219,27 +220,27 @@ checkTopLevelIds checks that Main.main or Main.mainPrimIO has correct type. \begin{code} checkTopLevelIds :: FAST_STRING -> TcEnv s -> TcM s () checkTopLevelIds mod final_env - = if (mod /= SLIT("Main")) then - returnTc () - else - case (lookupUFM_Directly localids mainIdKey, - lookupUFM_Directly localids mainPrimIOIdKey) of + | mod /= SLIT("Main") + = returnTc () + + | otherwise + = tcSetEnv final_env ( + tcLookupLocalValueByKey mainIdKey `thenNF_Tc` \ maybe_main -> + tcLookupLocalValueByKey mainPrimIOIdKey `thenNF_Tc` \ maybe_prim -> + tcLookupTyConByKey iOTyConKey `thenNF_Tc` \ io_tc -> + + case (maybe_main, maybe_prim) of (Just main, Nothing) -> tcAddErrCtxt mainCtxt $ - unifyTauTy ty_main (idType main) + unifyTauTy (applyTyCon io_tc [unitTy]) + (idType main) + (Nothing, Just prim) -> tcAddErrCtxt primCtxt $ - unifyTauTy ty_prim (idType prim) + unifyTauTy (mkPrimIoTy unitTy) + (idType prim) + (Just _ , Just _ ) -> failTc mainBothIdErr (Nothing, Nothing) -> failTc mainNoneIdErr - where - localids = getEnv_LocalIds final_env - tycons = getEnv_TyCons final_env - - io_tc = lookupWithDefaultUFM_Directly tycons io_panic iOTyConKey - io_panic = panic "TcModule: type IO not in scope" - - ty_main = applyTyCon io_tc [unitTy] - ty_prim = mkPrimIoTy unitTy - + ) mainCtxt sty = ppStr "main should have type IO ()" diff --git a/ghc/compiler/typecheck/TcMonoType.lhs b/ghc/compiler/typecheck/TcMonoType.lhs index 4ed8e502c0..91b1677a3b 100644 --- a/ghc/compiler/typecheck/TcMonoType.lhs +++ b/ghc/compiler/typecheck/TcMonoType.lhs @@ -18,7 +18,7 @@ import RnHsSyn ( RenamedPolyType(..), RenamedMonoType(..), import TcMonad import TcEnv ( tcLookupTyVar, tcLookupClass, tcLookupTyCon, - tcExtendTyVarEnv, tcTyVarScope + tcTyVarScope, tcTyVarScopeGivenKinds ) import TcKind ( TcKind, mkTcTypeKind, mkBoxedTypeKind, mkTcArrowKind, unifyKind, newKindVar, @@ -33,6 +33,7 @@ import TyVar ( GenTyVar, TyVar(..), mkTyVar ) import PrelInfo ( mkListTy, mkTupleTy ) import Type ( mkDictTy ) import Class ( cCallishClassKeys ) +import TyCon ( TyCon, Arity(..) ) import Unique ( Unique ) import Name ( Name(..), getNameShortName, isTyConName, getSynNameArity ) import PprStyle @@ -81,30 +82,33 @@ tcMonoTypeKind (MonoFunTy ty1 ty2) tcMonoTypeKind (MonoTyApp name tys) = mapAndUnzipTc tcMonoTypeKind tys `thenTc` \ (arg_kinds, arg_tys) -> - tc_mono_name name `thenNF_Tc` \ (fun_kind, fun_ty) -> + tc_mono_name name `thenNF_Tc` \ (fun_kind, maybe_arity, fun_ty) -> newKindVar `thenNF_Tc` \ result_kind -> unifyKind fun_kind (foldr mkTcArrowKind result_kind arg_kinds) `thenTc_` -- Check for saturated application in the special case of - -- type synoyms. Here the renamer has kindly attached the - -- arity to the Name. - synArityCheck name (length tys) `thenTc_` + -- type synoyms. + (case maybe_arity of + Just arity | arity /= n_args -> failTc (err arity) + other -> returnTc () + ) `thenTc_` returnTc (result_kind, foldl mkAppTy fun_ty arg_tys) + where + err arity = arityErr "Type synonym constructor" name arity n_args + n_args = length tys -- for unfoldings only: tcMonoTypeKind (MonoForAllTy tyvars_w_kinds ty) - = tcExtendTyVarEnv tyvar_names (tc_kinds `zip` tyvars) ( + = tcTyVarScopeGivenKinds names tc_kinds (\ tyvars -> tcMonoTypeKind ty `thenTc` \ (kind, ty') -> unifyKind kind mkTcTypeKind `thenTc_` returnTc (mkTcTypeKind, ty') ) where - (tyvar_names, kinds) = unzip tyvars_w_kinds - tyvars = zipWithEqual mk_tyvar tyvar_names kinds + (names, kinds) = unzip tyvars_w_kinds tc_kinds = map kindToTcKind kinds - mk_tyvar name kind = mkTyVar (getNameShortName name) (getItsUnique name) kind -- for unfoldings only: tcMonoTypeKind (MonoDictTy class_name ty) @@ -114,14 +118,14 @@ tcMonoTypeKind (MonoDictTy class_name ty) returnTc (mkTcTypeKind, mkDictTy clas arg_ty) -tc_mono_name :: Name -> NF_TcM s (TcKind s, Type) +tc_mono_name :: Name -> NF_TcM s (TcKind s, Maybe Arity, Type) tc_mono_name name@(Short _ _) -- Must be a type variable = tcLookupTyVar name `thenNF_Tc` \ (kind,tyvar) -> - returnNF_Tc (kind, mkTyVarTy tyvar) + returnNF_Tc (kind, Nothing, mkTyVarTy tyvar) tc_mono_name name | isTyConName name -- Must be a type constructor - = tcLookupTyCon name `thenNF_Tc` \ (kind,tycon) -> - returnNF_Tc (kind, mkTyConTy tycon) + = tcLookupTyCon name `thenNF_Tc` \ (kind,maybe_arity,tycon) -> + returnNF_Tc (kind, maybe_arity, mkTyConTy tycon) tc_mono_name name -- Renamer should have got it right = panic ("tc_mono_name:" ++ ppShow 1000 (ppr PprDebug name)) @@ -175,18 +179,6 @@ tcPolyType (HsForAllTy tyvar_names context ty) ) \end{code} -Auxilliary functions -~~~~~~~~~~~~~~~~~~~~ -\begin{code} -synArityCheck :: Name -> Int -> TcM s () -synArityCheck name n_args - = case getSynNameArity name of - Just arity | arity /= n_args -> failTc (err arity) - other -> returnTc () - where - err arity = arityErr "Type synonym constructor" name arity n_args -\end{code} - Errors and contexts ~~~~~~~~~~~~~~~~~~~ \begin{code} diff --git a/ghc/compiler/typecheck/TcTyClsDecls.lhs b/ghc/compiler/typecheck/TcTyClsDecls.lhs index 4e91011ba4..205c881f50 100644 --- a/ghc/compiler/typecheck/TcTyClsDecls.lhs +++ b/ghc/compiler/typecheck/TcTyClsDecls.lhs @@ -21,7 +21,7 @@ import TcMonad import Inst ( InstanceMapper(..) ) import TcClassDcl ( tcClassDecl1 ) import TcEnv ( tcExtendTyConEnv, tcExtendClassEnv, - tcExtendGlobalValEnv, tcExtendKindEnv, + tcExtendGlobalValEnv, tcTyVarScope, tcGetEnv ) import TcKind ( TcKind, newKindVars ) import TcTyDecls ( tcTyDecl ) @@ -82,14 +82,18 @@ Dealing with a group \begin{code} tcGroup :: InstanceMapper -> Bag Decl -> TcM s (TcEnv s) tcGroup inst_mapper decls - = fixTc ( \ ~(tycons,classes,_) -> + = pprTrace "tcGroup: " (ppCat (map (fst.fmt_decl) (bagToList decls))) $ - pprTrace "tcGroup: " (ppCat (map (fst.fmt_decl) (bagToList decls))) $ + -- TIE THE KNOT + fixTc ( \ ~(tycons,classes,_) -> -- EXTEND TYPE AND CLASS ENVIRONMENTS -- including their data constructors and class operations - tcExtendTyConEnv tycons $ - tcExtendClassEnv classes $ + -- NB: it's important that the tycons and classes come back in just + -- the same order from this fix as from get_binders, so that these + -- extend-env things work properly. A bit UGH-ish. + tcExtendTyConEnv tycon_names_w_arities tycons $ + tcExtendClassEnv class_names classes $ tcExtendGlobalValEnv (concat (map getTyConDataCons tycons)) $ tcExtendGlobalValEnv (concat (map getClassSelIds classes)) $ @@ -99,13 +103,6 @@ tcGroup inst_mapper decls -- DEAL WITH TYPE VARIABLES tcTyVarScope tyvar_names ( \ tyvars -> - -- MANUFACTURE NEW KINDS, AND EXTEND KIND ENV - newKindVars (length tycon_names) `thenNF_Tc` \ tycon_kinds -> - newKindVars (length class_names) `thenNF_Tc` \ class_kinds -> - tcExtendKindEnv tycon_names tycon_kinds $ - tcExtendKindEnv class_names class_kinds $ - - -- DEAL WITH THE DEFINITIONS THEMSELVES foldBag combine (tcDecl inst_mapper) (returnTc (emptyBag, emptyBag)) @@ -117,7 +114,7 @@ tcGroup inst_mapper decls returnTc final_env where - (tyvar_names, tycon_names, class_names) = get_binders decls + (tyvar_names, tycon_names_w_arities, class_names) = get_binders decls combine do_a do_b = do_a `thenTc` \ (a1,a2) -> @@ -238,6 +235,9 @@ set_name name = singletonUniqSet (getItsUnique name) set_to_bag set = listToBag (uniqSetToList set) \end{code} + +get_binders +~~~~~~~~~~~ Extract *binding* names from type and class decls. Type variables are bound in type, data, newtype and class declarations and the polytypes in the class op sigs. @@ -260,9 +260,9 @@ Monad c in bop's type signature means that D must have kind Type->Type. \begin{code} get_binders :: Bag Decl - -> ([Name], -- TyVars; no dups - [Name], -- Tycons; no dups - [Name]) -- Classes; no dups + -> ([Name], -- TyVars; no dups + [(Name, Maybe Arity)], -- Tycons; no dups; arities for synonyms + [Name]) -- Classes; no dups get_binders decls = (bagToList tyvars, bagToList tycons, bagToList classes) where @@ -274,21 +274,19 @@ get_binders decls = (bagToList tyvars, bagToList tycons, bagToList classes) = (a1 `unionBags` b1, a2 `unionBags` b2, a3 `unionBags` b3) get_binders1 (TyD (TyData _ name tyvars _ _ _ _)) - = (listToBag tyvars, unitBag name, emptyBag) + = (listToBag tyvars, unitBag (name,Nothing), emptyBag) get_binders1 (TyD (TyNew _ name tyvars _ _ _ _)) - = (listToBag tyvars, unitBag name, emptyBag) + = (listToBag tyvars, unitBag (name,Nothing), emptyBag) get_binders1 (TyD (TySynonym name tyvars _ _)) - = (listToBag tyvars, unitBag name, emptyBag) + = (listToBag tyvars, unitBag (name, Just (length tyvars)), emptyBag) get_binders1 (ClD (ClassDecl _ name tyvar sigs _ _ _)) = (unitBag tyvar `unionBags` sigs_tvs sigs, emptyBag, unitBag name) --- ToDo: will this duplicate the class tyvar - sigs_tvs sigs = unionManyBags (map sig_tvs sigs) where sig_tvs (ClassOpSig _ ty _ _) = pty_tvs ty - pty_tvs (HsForAllTy tvs _ _) = listToBag tvs + pty_tvs (HsForAllTy tvs _ _) = listToBag tvs -- tvs doesn't include the class tyvar \end{code} diff --git a/ghc/compiler/typecheck/TcTyDecls.lhs b/ghc/compiler/typecheck/TcTyDecls.lhs index 83a4c96732..9d6c08fe8d 100644 --- a/ghc/compiler/typecheck/TcTyDecls.lhs +++ b/ghc/compiler/typecheck/TcTyDecls.lhs @@ -46,7 +46,7 @@ tcTyDecl (TySynonym tycon_name tyvar_names rhs src_loc) tcAddErrCtxt (tySynCtxt tycon_name) $ -- Look up the pieces - tcLookupTyCon tycon_name `thenNF_Tc` \ (tycon_kind, rec_tycon) -> + tcLookupTyCon tycon_name `thenNF_Tc` \ (tycon_kind, _, rec_tycon) -> mapAndUnzipNF_Tc tcLookupTyVar tyvar_names `thenNF_Tc` \ (tyvar_kinds, rec_tyvars) -> -- Look at the rhs @@ -88,7 +88,7 @@ tcTyDataOrNew data_or_new context tycon_name tyvar_names con_decls derivings pra tcAddErrCtxt (tyDataCtxt tycon_name) $ -- Lookup the pieces - tcLookupTyCon tycon_name `thenNF_Tc` \ (tycon_kind, rec_tycon) -> + tcLookupTyCon tycon_name `thenNF_Tc` \ (tycon_kind, _, rec_tycon) -> mapAndUnzipNF_Tc tcLookupTyVar tyvar_names `thenNF_Tc` \ (tyvar_kinds, rec_tyvars) -> tc_derivs derivings `thenNF_Tc` \ derived_classes -> diff --git a/ghc/compiler/typecheck/TcType.lhs b/ghc/compiler/typecheck/TcType.lhs index ed2794dc17..1008e0cad8 100644 --- a/ghc/compiler/typecheck/TcType.lhs +++ b/ghc/compiler/typecheck/TcType.lhs @@ -132,13 +132,13 @@ tcInstType tenv ty_to_inst do env ty `thenNF_Tc` \ ty' -> returnNF_Tc (SynTy tycon tys' ty') - do env (FunTy arg res usage) = do env arg `thenNF_Tc` \ arg' -> - do env res `thenNF_Tc` \ res' -> - returnNF_Tc (FunTy arg' res' usage) + do env (FunTy arg res usage) = do env arg `thenNF_Tc` \ arg' -> + do env res `thenNF_Tc` \ res' -> + returnNF_Tc (FunTy arg' res' usage) - do env (AppTy fun arg) = do env fun `thenNF_Tc` \ fun' -> - do env arg `thenNF_Tc` \ arg' -> - returnNF_Tc (AppTy fun' arg') + do env (AppTy fun arg) = do env fun `thenNF_Tc` \ fun' -> + do env arg `thenNF_Tc` \ arg' -> + returnNF_Tc (AppTy fun' arg') do env (DictTy clas ty usage)= do env ty `thenNF_Tc` \ ty' -> returnNF_Tc (DictTy clas ty' usage) diff --git a/ghc/compiler/typecheck/Typecheck.lhs b/ghc/compiler/typecheck/Typecheck.lhs index f86c7dee49..d1893e3c54 100644 --- a/ghc/compiler/typecheck/Typecheck.lhs +++ b/ghc/compiler/typecheck/Typecheck.lhs @@ -50,10 +50,10 @@ typecheckModule [(Id, TypecheckedHsExpr)] -- constant instance binds ), - ([RenamedFixityDecl], [Id], UniqFM TyCon, UniqFM Class, Bag InstInfo), + ([RenamedFixityDecl], [Id], [TyCon], [Class], Bag InstInfo), -- things for the interface generator - (UniqFM TyCon, UniqFM Class), + ([TyCon], [Class]), -- environments of info from this module only FiniteMap TyCon [(Bool, [Maybe Type])], diff --git a/ghc/compiler/types/TyVar.lhs b/ghc/compiler/types/TyVar.lhs index a448f565b4..c963c1df4e 100644 --- a/ghc/compiler/types/TyVar.lhs +++ b/ghc/compiler/types/TyVar.lhs @@ -15,8 +15,10 @@ module TyVar ( growTyVarEnvList, isNullTyVarEnv, lookupTyVarEnv, GenTyVarSet(..), TyVarSet(..), - emptyTyVarSet, singletonTyVarSet, unionTyVarSets, tyVarListToSet, - tyVarSetToList, elementOfTyVarSet, minusTyVarSet, isEmptyTyVarSet + emptyTyVarSet, singletonTyVarSet, unionTyVarSets, + unionManyTyVarSets, intersectTyVarSets, mkTyVarSet, + tyVarSetToList, elementOfTyVarSet, minusTyVarSet, + isEmptyTyVarSet ) where CHK_Ubiq() -- debugging consistency check @@ -27,11 +29,10 @@ import Usage ( GenUsage, Usage(..), usageOmega ) import Kind ( Kind, mkBoxedTypeKind ) -- others -import UniqSet ( uniqSetToList, emptyUniqSet, singletonUniqSet, minusUniqSet, - unionUniqSets, elementOfUniqSet, isEmptyUniqSet, mkUniqSet, - UniqSet(..) ) +import UniqSet -- nearly all of it import UniqFM ( emptyUFM, listToUFM, addToUFM, lookupUFM, - plusUFM, sizeUFM, UniqFM ) + plusUFM, sizeUFM, UniqFM + ) import Maybes ( Maybe(..) ) import NameTypes ( ShortName ) import Pretty ( Pretty(..), PrettyRep, ppBeside, ppPStr ) @@ -107,22 +108,26 @@ type GenTyVarSet flexi = UniqSet (GenTyVar flexi) type TyVarSet = UniqSet TyVar emptyTyVarSet :: GenTyVarSet flexi +intersectTyVarSets:: GenTyVarSet flexi -> GenTyVarSet flexi -> GenTyVarSet flexi unionTyVarSets :: GenTyVarSet flexi -> GenTyVarSet flexi -> GenTyVarSet flexi +unionManyTyVarSets:: [GenTyVarSet flexi] -> GenTyVarSet flexi tyVarSetToList :: GenTyVarSet flexi -> [GenTyVar flexi] singletonTyVarSet :: GenTyVar flexi -> GenTyVarSet flexi elementOfTyVarSet :: GenTyVar flexi -> GenTyVarSet flexi -> Bool minusTyVarSet :: GenTyVarSet flexi -> GenTyVarSet flexi -> GenTyVarSet flexi isEmptyTyVarSet :: GenTyVarSet flexi -> Bool -tyVarListToSet :: [GenTyVar flexi] -> GenTyVarSet flexi +mkTyVarSet :: [GenTyVar flexi] -> GenTyVarSet flexi emptyTyVarSet = emptyUniqSet singletonTyVarSet = singletonUniqSet +intersectTyVarSets= intersectUniqSets unionTyVarSets = unionUniqSets +unionManyTyVarSets= unionManyUniqSets tyVarSetToList = uniqSetToList elementOfTyVarSet = elementOfUniqSet minusTyVarSet = minusUniqSet isEmptyTyVarSet = isEmptyUniqSet -tyVarListToSet = mkUniqSet +mkTyVarSet = mkUniqSet \end{code} Instance delarations diff --git a/ghc/compiler/types/Type.lhs b/ghc/compiler/types/Type.lhs index a6a6d679cd..a635130c12 100644 --- a/ghc/compiler/types/Type.lhs +++ b/ghc/compiler/types/Type.lhs @@ -3,7 +3,8 @@ module Type ( GenType(..), Type(..), TauType(..), - mkTyVarTy, getTyVar, getTyVar_maybe, isTyVarTy, + mkTyVarTy, mkTyVarTys, + getTyVar, getTyVar_maybe, isTyVarTy, mkAppTy, mkAppTys, splitAppTy, mkFunTy, mkFunTys, splitFunTy, getFunTy_maybe, mkTyConTy, getTyCon_maybe, applyTyCon, @@ -154,19 +155,20 @@ expandTy ty = ty Simple construction and analysis functions ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ \begin{code} -mkTyVarTy :: t -> GenType t u -mkTyVarTy = TyVarTy --- could we use something for (map mkTyVarTy blahs) ?? WDP +mkTyVarTy :: t -> GenType t u +mkTyVarTys :: [t] -> [GenType t y] +mkTyVarTy = TyVarTy +mkTyVarTys = map mkTyVarTy -- a common use of mkTyVarTy getTyVar :: String -> GenType t u -> t -getTyVar msg (TyVarTy tv) = tv -getTyVar msg (SynTy _ _ t) = getTyVar msg t -getTyVar msg other = error ("getTyVar" ++ msg) +getTyVar msg (TyVarTy tv) = tv +getTyVar msg (SynTy _ _ t) = getTyVar msg t +getTyVar msg other = panic ("getTyVar: " ++ msg) getTyVar_maybe :: GenType t u -> Maybe t -getTyVar_maybe (TyVarTy tv) = Just tv +getTyVar_maybe (TyVarTy tv) = Just tv getTyVar_maybe (SynTy _ _ t) = getTyVar_maybe t -getTyVar_maybe other = Nothing +getTyVar_maybe other = Nothing isTyVarTy :: GenType t u -> Bool isTyVarTy (TyVarTy tv) = True diff --git a/ghc/compiler/utils/Ubiq.lhi b/ghc/compiler/utils/Ubiq.lhi index 20e54b347d..b5783eebe3 100644 --- a/ghc/compiler/utils/Ubiq.lhi +++ b/ghc/compiler/utils/Ubiq.lhi @@ -21,7 +21,7 @@ import HsPragmas ( ClassOpPragmas, ClassPragmas, DataPragmas, GenPragmas, InstancePragmas ) import Id ( StrictnessMark, GenId, Id(..) ) -import IdInfo ( IdInfo, OptIdInfo(..), DeforestInfo, Demand, StrictnessInfo, UpdateInfo ) +import IdInfo ( IdInfo, OptIdInfo(..), ArityInfo, DeforestInfo, Demand, StrictnessInfo, UpdateInfo ) import Kind ( Kind ) import Literal ( Literal ) import Maybes ( MaybeErr ) @@ -72,6 +72,7 @@ class Outputable a where -- used everywhere and (b) the compiler doesn't lose much -- optimisation-wise by not seeing their pragma-gunk. +data ArityInfo data Bag a data BinderInfo data ClassOpPragmas a |