diff options
| author | Simon Peyton Jones <simonpj@microsoft.com> | 2011-05-06 15:56:06 +0100 | 
|---|---|---|
| committer | Simon Peyton Jones <simonpj@microsoft.com> | 2011-05-06 15:56:06 +0100 | 
| commit | ca53c38335cdc671f0b1e0949aa1514fc3fd72a5 (patch) | |
| tree | fb8adff51bd236c6c912a062702a8e08937333d4 | |
| parent | 4f8d714962667c219de4e684fe069136a0f78729 (diff) | |
| parent | 246183c669a1e851ccc42697dbbf309292bf2a08 (diff) | |
| download | haskell-ca53c38335cdc671f0b1e0949aa1514fc3fd72a5.tar.gz | |
Merge master into the ghc-new-co branch
67 files changed, 2616 insertions, 3640 deletions
| @@ -45,7 +45,7 @@ endif  include mk/custom-settings.mk  # No need to update makefiles for these targets: -REALGOALS=$(filter-out binary-dist binary-dist-prep bootstrapping-files framework-pkg clean clean_% distclean maintainer-clean show help install-docs test fulltest,$(MAKECMDGOALS)) +REALGOALS=$(filter-out binary-dist binary-dist-prep bootstrapping-files framework-pkg clean clean_% distclean maintainer-clean show help test fulltest,$(MAKECMDGOALS))  # configure touches certain files even if they haven't changed.  This  # can mean a lot of unnecessary recompilation after a re-configure, so @@ -102,12 +102,6 @@ framework-pkg:  	$(MAKE) -C distrib/MacOS $@  endif -# install-docs is a historical target that isn't supported in GHC 6.12. See #3662. -install-docs: -	@echo "The install-docs target is not supported in GHC 6.12.1 and later." -	@echo "'make install' now installs everything, including documentation." -	@exit 1 -  # If the user says 'make A B', then we don't want to invoke two  # instances of the rule above in parallel:  .NOTPARALLEL: diff --git a/aclocal.m4 b/aclocal.m4 index 7433873279..c7aba3e6a3 100644 --- a/aclocal.m4 +++ b/aclocal.m4 @@ -1031,18 +1031,6 @@ AC_SUBST([FopCmd])  ])# FP_PROG_FOP -# FP_PROG_HSTAGS -# ---------------- -# Sets the output variable HstagsCmd to the full Haskell tags program path. -# HstagsCmd is empty if no such program could be found. -AC_DEFUN([FP_PROG_HSTAGS], -[AC_PATH_PROG([HstagsCmd], [hasktags]) -if test -z "$HstagsCmd"; then -  AC_MSG_WARN([cannot find hasktags in your PATH, you will not be able to build the tags]) -fi -])# FP_PROG_HSTAGS - -  # FP_PROG_GHC_PKG  # ----------------  # Try to find a ghc-pkg matching the ghc mentioned in the environment variable diff --git a/compiler/cmm/CLabel.hs b/compiler/cmm/CLabel.hs index 901b13b342..3451c7d5a9 100644 --- a/compiler/cmm/CLabel.hs +++ b/compiler/cmm/CLabel.hs @@ -101,7 +101,7 @@ module CLabel (          hasCAF,  	infoLblToEntryLbl, entryLblToInfoLbl, cvtToClosureLbl, cvtToSRTLbl,  	needsCDecl, isAsmTemp, maybeAsmTemp, externallyVisibleCLabel, -        isMathFun, isCas, +        isMathFun,   	isCFunctionLabel, isGcPtrLabel, labelDynamic,  	pprCLabel @@ -590,14 +590,6 @@ maybeAsmTemp (AsmTempLabel uq) 		= Just uq  maybeAsmTemp _ 	    	       		= Nothing --- | Check whether a label corresponds to our cas function. ---      We #include the prototype for this, so we need to avoid ---      generating out own C prototypes. -isCas :: CLabel -> Bool -isCas (CmmLabel pkgId fn _) = pkgId == rtsPackageId && fn == fsLit "cas" -isCas _                     = False - -  -- | Check whether a label corresponds to a C function that has   --      a prototype in a system header somehere, or is built-in  --      to the C compiler. For these labels we avoid generating our @@ -858,8 +850,8 @@ instance Outputable CLabel where  pprCLabel :: CLabel -> SDoc -#if ! OMIT_NATIVE_CODEGEN  pprCLabel (AsmTempLabel u) + | cGhcWithNativeCodeGen == "YES"    =  getPprStyle $ \ sty ->       if asmStyle sty then   	ptext asmTempLabelPrefix <> pprUnique u @@ -867,23 +859,22 @@ pprCLabel (AsmTempLabel u)  	char '_' <> pprUnique u  pprCLabel (DynamicLinkerLabel info lbl) + | cGhcWithNativeCodeGen == "YES"     = pprDynamicLinkerAsmLabel info lbl  pprCLabel PicBaseLabel + | cGhcWithNativeCodeGen == "YES"     = ptext (sLit "1b")  pprCLabel (DeadStripPreventer lbl) + | cGhcWithNativeCodeGen == "YES"     = pprCLabel lbl <> ptext (sLit "_dsp") -#endif -pprCLabel lbl =  -#if ! OMIT_NATIVE_CODEGEN -    getPprStyle $ \ sty -> -    if asmStyle sty then  -	maybe_underscore (pprAsmCLbl lbl) -    else -#endif -       pprCLbl lbl +pprCLabel lbl +   = getPprStyle $ \ sty -> +     if cGhcWithNativeCodeGen == "YES" && asmStyle sty +     then maybe_underscore (pprAsmCLbl lbl) +     else pprCLbl lbl  maybe_underscore doc    | underscorePrefix = pp_cSEP <> doc diff --git a/compiler/cmm/CmmOpt.hs b/compiler/cmm/CmmOpt.hs index c71f188ba7..a2eecd5c48 100644 --- a/compiler/cmm/CmmOpt.hs +++ b/compiler/cmm/CmmOpt.hs @@ -14,6 +14,7 @@  -----------------------------------------------------------------------------  module CmmOpt ( +	cmmEliminateDeadBlocks,  	cmmMiniInline,  	cmmMachOpFold,  	cmmLoopifyForC, @@ -30,10 +31,69 @@ import UniqFM  import Unique  import FastTypes  import Outputable +import BlockId  import Data.Bits  import Data.Word  import Data.Int +import Data.Maybe +import Data.List + +import Compiler.Hoopl hiding (Unique) + +-- ----------------------------------------------------------------------------- +-- Eliminates dead blocks + +{- +We repeatedly expand the set of reachable blocks until we hit a +fixpoint, and then prune any blocks that were not in this set.  This is +actually a required optimization, as dead blocks can cause problems +for invariants in the linear register allocator (and possibly other +places.) +-} + +-- Deep fold over statements could probably be abstracted out, but it +-- might not be worth the effort since OldCmm is moribund +cmmEliminateDeadBlocks :: [CmmBasicBlock] -> [CmmBasicBlock] +cmmEliminateDeadBlocks [] = [] +cmmEliminateDeadBlocks blocks@(BasicBlock base_id _:_) = +    let -- Calculate what's reachable from what block +        reachableMap = foldl' f emptyUFM blocks -- lazy in values +            where f m (BasicBlock block_id stmts) = addToUFM m block_id (reachableFrom stmts) +        reachableFrom stmts = foldl stmt [] stmts +            where +                stmt m CmmNop = m +                stmt m (CmmComment _) = m +                stmt m (CmmAssign _ e) = expr m e +                stmt m (CmmStore e1 e2) = expr (expr m e1) e2 +                stmt m (CmmCall c _ as _ _) = f (actuals m as) c +                    where f m (CmmCallee e _) = expr m e +                          f m (CmmPrim _) = m +                stmt m (CmmBranch b) = b:m +                stmt m (CmmCondBranch e b) = b:(expr m e) +                stmt m (CmmSwitch e bs) = catMaybes bs ++ expr m e +                stmt m (CmmJump e as) = expr (actuals m as) e +                stmt m (CmmReturn as) = actuals m as +                actuals m as = foldl' (\m h -> expr m (hintlessCmm h)) m as +                -- We have to do a deep fold into CmmExpr because +                -- there may be a BlockId in the CmmBlock literal. +                expr m (CmmLit l) = lit m l +                expr m (CmmLoad e _) = expr m e +                expr m (CmmReg _) = m +                expr m (CmmMachOp _ es) = foldl' expr m es +                expr m (CmmStackSlot _ _) = m +                expr m (CmmRegOff _ _) = m +                lit m (CmmBlock b) = b:m +                lit m _ = m +        -- go todo done +        reachable = go [base_id] (setEmpty :: BlockSet) +          where go []     m = m +                go (x:xs) m +                    | setMember x m = go xs m +                    | otherwise     = go (add ++ xs) (setInsert x m) +                        where add = fromMaybe (panic "cmmEliminateDeadBlocks: unknown block") +                                              (lookupUFM reachableMap x) +    in filter (\(BasicBlock block_id _) -> setMember block_id reachable) blocks  -- -----------------------------------------------------------------------------  -- The mini-inliner diff --git a/compiler/cmm/PprC.hs b/compiler/cmm/PprC.hs index d363cef50b..10f4e8bacf 100644 --- a/compiler/cmm/PprC.hs +++ b/compiler/cmm/PprC.hs @@ -248,7 +248,7 @@ pprStmt stmt = case stmt of                  | CmmNeverReturns <- ret ->                      let myCall = pprCall (pprCLabel lbl) cconv results args safety                      in (real_fun_proto lbl, myCall) -                | not (isMathFun lbl || isCas lbl) -> +                | not (isMathFun lbl) ->                      let myCall = braces (                                       pprCFunType (char '*' <> text "ghcFunPtr") cconv results args <> semi                                    $$ text "ghcFunPtr" <+> equals <+> cast_fn <> semi diff --git a/compiler/deSugar/Check.lhs b/compiler/deSugar/Check.lhs index bcbf4435eb..cc00536e85 100644 --- a/compiler/deSugar/Check.lhs +++ b/compiler/deSugar/Check.lhs @@ -111,7 +111,8 @@ check :: [EquationInfo] -> ([ExhaustivePat], [EquationInfo])    -- if there are view patterns, just give up - don't know what the function is  check qs = (untidy_warns, shadowed_eqns)        where -	(warns, used_nos) = check' ([1..] `zip` map tidy_eqn qs) +        tidy_qs = map tidy_eqn qs +	(warns, used_nos) = check' ([1..] `zip` tidy_qs)  	untidy_warns = map untidy_exhaustive warns   	shadowed_eqns = [eqn | (eqn,i) <- qs `zip` [1..],   				not (i `elementOfUniqSet` used_nos)] @@ -670,8 +671,6 @@ tidy_pat (CoPat _ pat _)  = tidy_pat pat  tidy_pat (NPlusKPat id _ _ _) = WildPat (idType (unLoc id))  tidy_pat (ViewPat _ _ ty)     = WildPat ty -tidy_pat (NPat lit mb_neg eq) = tidyNPat lit mb_neg eq -  tidy_pat pat@(ConPatOut { pat_con = L _ id, pat_args = ps })    = pat { pat_args = tidy_con id ps } @@ -695,16 +694,18 @@ tidy_pat (TuplePat ps boxity ty)    where      arity = length ps --- Unpack string patterns fully, so we can see when they overlap with --- each other, or even explicit lists of Chars. -tidy_pat (LitPat lit) +tidy_pat (NPat lit mb_neg eq) = tidyNPat tidy_lit_pat lit mb_neg eq +tidy_pat (LitPat lit)         = tidy_lit_pat lit + +tidy_lit_pat :: HsLit -> Pat Id +-- Unpack string patterns fully, so we can see when they  +-- overlap with each other, or even explicit lists of Chars. +tidy_lit_pat lit    | HsString s <- lit -  = unLoc $ foldr (\c pat -> mkPrefixConPat consDataCon [mk_char_lit c, pat] stringTy) +  = unLoc $ foldr (\c pat -> mkPrefixConPat consDataCon [mkCharLitPat c, pat] stringTy)  		  (mkPrefixConPat nilDataCon [] stringTy) (unpackFS s)    | otherwise    = tidyLitPat lit  -  where -    mk_char_lit c = mkPrefixConPat charDataCon [nlLitPat (HsCharPrim c)] charTy  -----------------  tidy_con :: DataCon -> HsConPatDetails Id -> HsConPatDetails Id diff --git a/compiler/deSugar/Coverage.lhs b/compiler/deSugar/Coverage.lhs index 0daa6befc4..8071da756f 100644 --- a/compiler/deSugar/Coverage.lhs +++ b/compiler/deSugar/Coverage.lhs @@ -301,10 +301,9 @@ addTickHsExpr (HsLet binds e) =  	liftM2 HsLet  		(addTickHsLocalBinds binds) -- to think about: !patterns.                  (addTickLHsExprNeverOrAlways e) -addTickHsExpr (HsDo cxt stmts last_exp srcloc) = do -        (stmts', last_exp') <- addTickLStmts' forQual stmts  -                                     (addTickLHsExpr last_exp) -	return (HsDo cxt stmts' last_exp' srcloc) +addTickHsExpr (HsDo cxt stmts srcloc)  +  = do { (stmts', _) <- addTickLStmts' forQual stmts (return ()) +       ; return (HsDo cxt stmts' srcloc) }    where  	forQual = case cxt of  		    ListComp -> Just $ BinBox QualBinBox @@ -424,45 +423,50 @@ addTickLStmts isGuard stmts = do  addTickLStmts' :: (Maybe (Bool -> BoxLabel)) -> [LStmt Id] -> TM a                  -> TM ([LStmt Id], a)  addTickLStmts' isGuard lstmts res -  = bindLocals binders $ do -        lstmts' <- mapM (liftL (addTickStmt isGuard)) lstmts -        a <- res -        return (lstmts', a) -  where -        binders = collectLStmtsBinders lstmts +  = bindLocals (collectLStmtsBinders lstmts) $  +    do { lstmts' <- mapM (liftL (addTickStmt isGuard)) lstmts +       ; a <- res +       ; return (lstmts', a) }  addTickStmt :: (Maybe (Bool -> BoxLabel)) -> Stmt Id -> TM (Stmt Id) +addTickStmt _isGuard (LastStmt e ret) = do +	liftM2 LastStmt +		(addTickLHsExpr e) +		(addTickSyntaxExpr hpcSrcSpan ret)  addTickStmt _isGuard (BindStmt pat e bind fail) = do  	liftM4 BindStmt  		(addTickLPat pat)  		(addTickLHsExprAlways e)  		(addTickSyntaxExpr hpcSrcSpan bind)  		(addTickSyntaxExpr hpcSrcSpan fail) -addTickStmt isGuard (ExprStmt e bind' ty) = do -	liftM3 ExprStmt +addTickStmt isGuard (ExprStmt e bind' guard' ty) = do +	liftM4 ExprStmt  		(addTick isGuard e)  		(addTickSyntaxExpr hpcSrcSpan bind') +		(addTickSyntaxExpr hpcSrcSpan guard')  		(return ty)  addTickStmt _isGuard (LetStmt binds) = do  	liftM LetStmt  		(addTickHsLocalBinds binds) -addTickStmt isGuard (ParStmt pairs) = do -    liftM ParStmt  +addTickStmt isGuard (ParStmt pairs mzipExpr bindExpr returnExpr) = do +    liftM4 ParStmt           (mapM (addTickStmtAndBinders isGuard) pairs) - -addTickStmt isGuard (TransformStmt stmts ids usingExpr maybeByExpr) = do -    liftM4 TransformStmt  -        (addTickLStmts isGuard stmts) -        (return ids) -        (addTickLHsExprAlways usingExpr) -        (addTickMaybeByLHsExpr maybeByExpr) - -addTickStmt isGuard (GroupStmt stmts binderMap by using) = do -    liftM4 GroupStmt  -        (addTickLStmts isGuard stmts) -        (return binderMap) -        (fmapMaybeM  addTickLHsExprAlways by) -	(fmapEitherM addTickLHsExprAlways (addTickSyntaxExpr hpcSrcSpan) using) +        (addTickSyntaxExpr hpcSrcSpan mzipExpr) +        (addTickSyntaxExpr hpcSrcSpan bindExpr) +        (addTickSyntaxExpr hpcSrcSpan returnExpr) + +addTickStmt isGuard stmt@(TransStmt { trS_stmts = stmts +                                    , trS_by = by, trS_using = using +                                    , trS_ret = returnExpr, trS_bind = bindExpr +                                    , trS_fmap = liftMExpr }) = do +    t_s <- addTickLStmts isGuard stmts +    t_y <- fmapMaybeM  addTickLHsExprAlways by +    t_u <- addTickLHsExprAlways using +    t_f <- addTickSyntaxExpr hpcSrcSpan returnExpr +    t_b <- addTickSyntaxExpr hpcSrcSpan bindExpr +    t_m <- addTickSyntaxExpr hpcSrcSpan liftMExpr +    return $ stmt { trS_stmts = t_s, trS_by = t_y, trS_using = t_u +                  , trS_ret = t_f, trS_bind = t_b, trS_fmap = t_m }  addTickStmt isGuard stmt@(RecStmt {})    = do { stmts' <- addTickLStmts isGuard (recS_stmts stmt) @@ -483,12 +487,6 @@ addTickStmtAndBinders isGuard (stmts, ids) =          (addTickLStmts isGuard stmts)          (return ids) -addTickMaybeByLHsExpr :: Maybe (LHsExpr Id) -> TM (Maybe (LHsExpr Id)) -addTickMaybeByLHsExpr maybeByExpr =  -    case maybeByExpr of -        Nothing -> return Nothing -        Just byExpr -> addTickLHsExprAlways byExpr >>= (return . Just) -  addTickHsLocalBinds :: HsLocalBinds Id -> TM (HsLocalBinds Id)  addTickHsLocalBinds (HsValBinds binds) =   	liftM HsValBinds  @@ -569,9 +567,9 @@ addTickHsCmd (HsLet binds c) =  	liftM2 HsLet  		(addTickHsLocalBinds binds) -- to think about: !patterns.                  (addTickLHsCmd c) -addTickHsCmd (HsDo cxt stmts last_exp srcloc) = do -        (stmts', last_exp') <- addTickLCmdStmts' stmts (addTickLHsCmd last_exp) -	return (HsDo cxt stmts' last_exp' srcloc) +addTickHsCmd (HsDo cxt stmts srcloc) +  = do { (stmts', _) <- addTickLCmdStmts' stmts (return ()) +       ; return (HsDo cxt stmts' srcloc) }  addTickHsCmd (HsArrApp	 e1 e2 ty1 arr_ty lr) =           liftM5 HsArrApp @@ -635,10 +633,15 @@ addTickCmdStmt (BindStmt pat c bind fail) = do  		(addTickLHsCmd c)  		(return bind)  		(return fail) -addTickCmdStmt (ExprStmt c bind' ty) = do -	liftM3 ExprStmt +addTickCmdStmt (LastStmt c ret) = do +	liftM2 LastStmt +		(addTickLHsCmd c) +		(addTickSyntaxExpr hpcSrcSpan ret) +addTickCmdStmt (ExprStmt c bind' guard' ty) = do +	liftM4 ExprStmt  		(addTickLHsCmd c) -		(return bind') +		(addTickSyntaxExpr hpcSrcSpan bind') +                (addTickSyntaxExpr hpcSrcSpan guard')  		(return ty)  addTickCmdStmt (LetStmt binds) = do  	liftM LetStmt diff --git a/compiler/deSugar/DsArrows.lhs b/compiler/deSugar/DsArrows.lhs index 58bf6b88e7..a5bf2b69d6 100644 --- a/compiler/deSugar/DsArrows.lhs +++ b/compiler/deSugar/DsArrows.lhs @@ -541,8 +541,8 @@ dsCmd ids local_vars env_ids stack res_ty (HsLet binds body) = do                          core_body,          exprFreeVars core_binds `intersectVarSet` local_vars) -dsCmd ids local_vars env_ids [] res_ty (HsDo _ctxt stmts body _) -  = dsCmdDo ids local_vars env_ids res_ty stmts body +dsCmd ids local_vars env_ids [] res_ty (HsDo _ctxt stmts _) +  = dsCmdDo ids local_vars env_ids res_ty stmts   --	A |- e :: forall e. a1 (e*ts1) t1 -> ... an (e*tsn) tn -> a (e*ts) t  --	A | xs |- ci :: [tsi] ti @@ -618,7 +618,6 @@ dsCmdDo :: DsCmdEnv		-- arrow combinators  				-- so don't pull on it too early  	-> Type			-- return type of the statement  	-> [LStmt Id]		-- statements to desugar -	-> LHsExpr Id		-- body  	-> DsM (CoreExpr,	-- desugared expression  		IdSet)		-- set of local vars that occur free @@ -626,15 +625,17 @@ dsCmdDo :: DsCmdEnv		-- arrow combinators  --	--------------------------  --	A | xs |- do { c } :: [] t -dsCmdDo ids local_vars env_ids res_ty [] body +dsCmdDo _ _ _ _ [] = panic "dsCmdDo" + +dsCmdDo ids local_vars env_ids res_ty [L _ (LastStmt body _)]    = dsLCmd ids local_vars env_ids [] res_ty body -dsCmdDo ids local_vars env_ids res_ty (stmt:stmts) body = do +dsCmdDo ids local_vars env_ids res_ty (stmt:stmts) = do      let          bound_vars = mkVarSet (collectLStmtBinders stmt)          local_vars' = local_vars `unionVarSet` bound_vars      (core_stmts, _, env_ids') <- fixDs (\ ~(_,_,env_ids') -> do -        (core_stmts, fv_stmts) <- dsCmdDo ids local_vars' env_ids' res_ty stmts body +        (core_stmts, fv_stmts) <- dsCmdDo ids local_vars' env_ids' res_ty stmts           return (core_stmts, fv_stmts, varSetElems fv_stmts))      (core_stmt, fv_stmt) <- dsCmdLStmt ids local_vars env_ids env_ids' stmt      return (do_compose ids @@ -674,7 +675,7 @@ dsCmdStmt  --		---> arr (\ (xs) -> ((xs1),(xs'))) >>> first c >>>  --			arr snd >>> ss -dsCmdStmt ids local_vars env_ids out_ids (ExprStmt cmd _ c_ty) = do +dsCmdStmt ids local_vars env_ids out_ids (ExprStmt cmd _ _ c_ty) = do      (core_cmd, fv_cmd, env_ids1) <- dsfixCmd ids local_vars [] c_ty cmd      core_mux <- matchEnvStack env_ids []          (mkCorePairExpr (mkBigCoreVarTup env_ids1) (mkBigCoreVarTup out_ids)) diff --git a/compiler/deSugar/DsExpr.lhs b/compiler/deSugar/DsExpr.lhs index 5db2175a50..e33b113ae7 100644 --- a/compiler/deSugar/DsExpr.lhs +++ b/compiler/deSugar/DsExpr.lhs @@ -325,26 +325,12 @@ dsExpr (HsLet binds body) = do  -- We need the `ListComp' form to use `deListComp' (rather than the "do" form)  -- because the interpretation of `stmts' depends on what sort of thing it is.  -- -dsExpr (HsDo ListComp stmts body result_ty) -  =	-- Special case for list comprehensions -    dsListComp stmts body elt_ty -  where -    [elt_ty] = tcTyConAppArgs result_ty - -dsExpr (HsDo DoExpr stmts body result_ty) -  = dsDo stmts body result_ty - -dsExpr (HsDo GhciStmt stmts body result_ty) -  = dsDo stmts body result_ty - -dsExpr (HsDo MDoExpr stmts body result_ty) -  = dsDo stmts body result_ty - -dsExpr (HsDo PArrComp stmts body result_ty) -  =	-- Special case for array comprehensions -    dsPArrComp (map unLoc stmts) body elt_ty -  where -    [elt_ty] = tcTyConAppArgs result_ty +dsExpr (HsDo ListComp  stmts res_ty) = dsListComp stmts res_ty +dsExpr (HsDo PArrComp  stmts _)      = dsPArrComp (map unLoc stmts) +dsExpr (HsDo DoExpr    stmts _)      = dsDo stmts  +dsExpr (HsDo GhciStmt  stmts _)      = dsDo stmts  +dsExpr (HsDo MDoExpr   stmts _)      = dsDo stmts  +dsExpr (HsDo MonadComp stmts _)      = dsMonadComp stmts  dsExpr (HsIf mb_fun guard_expr then_expr else_expr)    = do { pred <- dsLExpr guard_expr @@ -708,25 +694,20 @@ handled in DsListComp).  Basically does the translation given in the  Haskell 98 report:  \begin{code} -dsDo	:: [LStmt Id] -	-> LHsExpr Id -	-> Type			-- Type of the whole expression -	-> DsM CoreExpr - -dsDo stmts body result_ty +dsDo :: [LStmt Id] -> DsM CoreExpr +dsDo stmts    = goL stmts    where -    -- result_ty must be of the form (m b) -    (m_ty, _b_ty) = tcSplitAppTy result_ty - -    goL [] = dsLExpr body -    goL ((L loc stmt):lstmts) = putSrcSpanDs loc (go loc stmt lstmts) +    goL [] = panic "dsDo" +    goL (L loc stmt:lstmts) = putSrcSpanDs loc (go loc stmt lstmts) -    go _ (ExprStmt rhs then_expr _) stmts +    go _ (LastStmt body _) stmts +      = ASSERT( null stmts ) dsLExpr body +        -- The 'return' op isn't used for 'do' expressions + +    go _ (ExprStmt rhs then_expr _ _) stmts        = do { rhs2 <- dsLExpr rhs -           ; case tcSplitAppTy_maybe (exprType rhs2) of -                Just (container_ty, returning_ty) -> warnDiscardedDoBindings rhs container_ty returning_ty -                _                                 -> return () +           ; warnDiscardedDoBindings rhs (exprType rhs2)              ; then_expr2 <- dsExpr then_expr  	   ; rest <- goL stmts  	   ; return (mkApps then_expr2 [rhs2, rest]) } @@ -750,29 +731,29 @@ dsDo stmts body result_ty      go loc (RecStmt { recS_stmts = rec_stmts, recS_later_ids = later_ids                      , recS_rec_ids = rec_ids, recS_ret_fn = return_op                      , recS_mfix_fn = mfix_op, recS_bind_fn = bind_op -                    , recS_rec_rets = rec_rets }) stmts +                    , recS_rec_rets = rec_rets, recS_ret_ty = body_ty }) stmts        = ASSERT( length rec_ids > 0 )          goL (new_bind_stmt : stmts)        where -        -- returnE <- dsExpr return_id -        -- mfixE <- dsExpr mfix_id -        new_bind_stmt = L loc $ BindStmt (mkLHsPatTup later_pats) mfix_app -                                         bind_op  +        new_bind_stmt = L loc $ BindStmt (mkLHsPatTup later_pats) +                                         mfix_app bind_op                                            noSyntaxExpr  -- Tuple cannot fail          tup_ids      = rec_ids ++ filterOut (`elem` rec_ids) later_ids +        tup_ty       = mkBoxedTupleTy (map idType tup_ids) -- Deals with singleton case          rec_tup_pats = map nlVarPat tup_ids          later_pats   = rec_tup_pats          rets         = map noLoc rec_rets - -        mfix_app   = nlHsApp (noLoc mfix_op) mfix_arg -        mfix_arg   = noLoc $ HsLam (MatchGroup [mkSimpleMatch [mfix_pat] body] -                                             (mkFunTy tup_ty body_ty)) -        mfix_pat   = noLoc $ LazyPat $ mkLHsPatTup rec_tup_pats -        body       = noLoc $ HsDo DoExpr rec_stmts return_app body_ty -        return_app = nlHsApp (noLoc return_op) (mkLHsTupleExpr rets) -	body_ty    = mkAppTy m_ty tup_ty -        tup_ty     = mkBoxedTupleTy (map idType tup_ids) -- Deals with singleton case +        mfix_app     = nlHsApp (noLoc mfix_op) mfix_arg +        mfix_arg     = noLoc $ HsLam (MatchGroup [mkSimpleMatch [mfix_pat] body] +                                                 (mkFunTy tup_ty body_ty)) +        mfix_pat     = noLoc $ LazyPat $ mkLHsPatTup rec_tup_pats +        body         = noLoc $ HsDo DoExpr (rec_stmts ++ [ret_stmt]) body_ty +        ret_app      = nlHsApp (noLoc return_op) (mkLHsTupleExpr rets) +        ret_stmt     = noLoc $ mkLastStmt ret_app +		     -- This LastStmt will be desugared with dsDo,  +		     -- which ignores the return_op in the LastStmt, +		     -- so we must apply the return_op explicitly   handle_failure :: LPat Id -> MatchResult -> SyntaxExpr Id -> DsM CoreExpr      -- In a do expression, pattern-match failure just calls @@ -790,104 +771,6 @@ mk_fail_msg pat = "Pattern match failure in do expression at " ++  		  showSDoc (ppr (getLoc pat))  \end{code} -Translation for RecStmt's:  ------------------------------ -We turn (RecStmt [v1,..vn] stmts) into: -   -  (v1,..,vn) <- mfix (\~(v1,..vn). do stmts -				      return (v1,..vn)) - -\begin{code} -{- -dsMDo   :: HsStmtContext Name -        -> [(Name,Id)] -	-> [LStmt Id] -	-> LHsExpr Id -	-> Type			-- Type of the whole expression -	-> DsM CoreExpr - -dsMDo ctxt tbl stmts body result_ty -  = goL stmts -  where -    goL [] = dsLExpr body -    goL ((L loc stmt):lstmts) = putSrcSpanDs loc (go loc stmt lstmts) -   -    (m_ty, b_ty) = tcSplitAppTy result_ty	-- result_ty must be of the form (m b) -    return_id = lookupEvidence tbl returnMName -    bind_id   = lookupEvidence tbl bindMName -    then_id   = lookupEvidence tbl thenMName -    fail_id   = lookupEvidence tbl failMName - -    go _ (LetStmt binds) stmts -      = do { rest <- goL stmts -	   ; dsLocalBinds binds rest } - -    go _ (ExprStmt rhs then_expr rhs_ty) stmts -      = do { rhs2 <- dsLExpr rhs -	   ; warnDiscardedDoBindings rhs m_ty rhs_ty -           ; then_expr2 <- dsExpr then_expr -           ; rest <- goL stmts -           ; return (mkApps then_expr2 [rhs2, rest]) } -     -    go _ (BindStmt pat rhs bind_op _) stmts -      = do { body     <- goL stmts -           ; rhs'     <- dsLExpr rhs -           ; bind_op' <- dsExpr bind_op -           ; var   <- selectSimpleMatchVarL pat -	   ; match <- matchSinglePat (Var var) (StmtCtxt ctxt) pat -                                     result_ty (cantFailMatchResult body) -           ; match_code <- handle_failure pat match fail_op -           ; return (mkApps bind_op [rhs', Lam var match_code]) } -     -    go loc (RecStmt { recS_stmts = rec_stmts, recS_later_ids = later_ids -                    , recS_rec_ids = rec_ids, recS_rec_rets = rec_rets -                    , recS_mfix_fn = mfix_op, recS_bind_fn = bind_op }) stmts -      = ASSERT( length rec_ids > 0 ) -        ASSERT( length rec_ids == length rec_rets ) -        ASSERT( isEmptyTcEvBinds _ev_binds ) -        pprTrace "dsMDo" (ppr later_ids) $ -	 goL (new_bind_stmt : stmts) -      where -        new_bind_stmt = L loc $ BindStmt (mk_tup_pat later_pats) mfix_app -                                         bind_op noSyntaxExpr -	 -		-- Remove the later_ids that appear (without fancy coercions)  -		-- in rec_rets, because there's no need to knot-tie them separately -		-- See Note [RecStmt] in HsExpr -	later_ids'   = filter (`notElem` mono_rec_ids) later_ids -	mono_rec_ids = [ id | HsVar id <- rec_rets ] -     -        mfix_app = nlHsApp (noLoc mfix_op) mfix_arg -	mfix_arg = noLoc $ HsLam (MatchGroup [mkSimpleMatch [mfix_pat] body] -					     (mkFunTy tup_ty body_ty)) - -	-- The rec_tup_pat must bind the rec_ids only; remember that the  -	-- 	trimmed_laters may share the same Names -	-- Meanwhile, the later_pats must bind the later_vars -	rec_tup_pats = map mk_wild_pat later_ids' ++ map nlVarPat rec_ids -	later_pats   = map nlVarPat    later_ids' ++ map mk_later_pat rec_ids -	rets         = map nlHsVar     later_ids' ++ map noLoc rec_rets - -	mfix_pat = noLoc $ LazyPat $ mk_tup_pat rec_tup_pats -	body     = noLoc $ HsDo ctxt rec_stmts return_app body_ty -	body_ty = mkAppTy m_ty tup_ty -	tup_ty  = mkBoxedTupleTy (map idType (later_ids' ++ rec_ids))  -- Deals with singleton case - -        return_app  = nlHsApp (noLoc return_op) (mkLHsTupleExpr rets) - -	mk_wild_pat :: Id -> LPat Id  -   	mk_wild_pat v = noLoc $ WildPat $ idType v - -	mk_later_pat :: Id -> LPat Id -	mk_later_pat v | v `elem` later_ids' = mk_wild_pat v -		       | otherwise	     = nlVarPat v - - 	mk_tup_pat :: [LPat Id] -> LPat Id -  	mk_tup_pat [p] = p -	mk_tup_pat ps  = noLoc $ mkVanillaTuplePat ps Boxed --} -\end{code} -  %************************************************************************  %*									* @@ -927,30 +810,34 @@ conversionNames  \begin{code}  -- Warn about certain types of values discarded in monadic bindings (#3263) -warnDiscardedDoBindings :: LHsExpr Id -> Type -> Type -> DsM () -warnDiscardedDoBindings rhs container_ty returning_ty = do { -          -- Warn about discarding non-() things in 'monadic' binding -        ; warn_unused <- doptDs Opt_WarnUnusedDoBind -        ; if warn_unused && not (returning_ty `eqType` unitTy) -           then warnDs (unusedMonadBind rhs returning_ty) -           else do { -          -- Warn about discarding m a things in 'monadic' binding of the same type, -          -- but only if we didn't already warn due to Opt_WarnUnusedDoBind -        ; warn_wrong <- doptDs Opt_WarnWrongDoBind -        ; case tcSplitAppTy_maybe returning_ty of -                  Just (returning_container_ty, _) -> when (warn_wrong && container_ty `eqType` returning_container_ty) $ -                                                            warnDs (wrongMonadBind rhs returning_ty) -                  _ -> return () } } +warnDiscardedDoBindings :: LHsExpr Id -> Type -> DsM () +warnDiscardedDoBindings rhs rhs_ty +  | Just (m_ty, elt_ty) <- tcSplitAppTy_maybe rhs_ty +  = do {  -- Warn about discarding non-() things in 'monadic' binding +       ; warn_unused <- doptDs Opt_WarnUnusedDoBind +       ; if warn_unused && not (isUnitTy elt_ty) +         then warnDs (unusedMonadBind rhs elt_ty) +         else  +         -- Warn about discarding m a things in 'monadic' binding of the same type, +         -- but only if we didn't already warn due to Opt_WarnUnusedDoBind +    do { warn_wrong <- doptDs Opt_WarnWrongDoBind +       ; case tcSplitAppTy_maybe elt_ty of +           Just (elt_m_ty, _) | warn_wrong, m_ty `eqType` elt_m_ty +                              -> warnDs (wrongMonadBind rhs elt_ty) +           _ -> return () } } + +  | otherwise	-- RHS does have type of form (m ty), which is wierd +  = return ()   -- but at lesat this warning is irrelevant  unusedMonadBind :: LHsExpr Id -> Type -> SDoc -unusedMonadBind rhs returning_ty -  = ptext (sLit "A do-notation statement discarded a result of type") <+> ppr returning_ty <> dot $$ +unusedMonadBind rhs elt_ty +  = ptext (sLit "A do-notation statement discarded a result of type") <+> ppr elt_ty <> dot $$      ptext (sLit "Suppress this warning by saying \"_ <- ") <> ppr rhs <> ptext (sLit "\",") $$      ptext (sLit "or by using the flag -fno-warn-unused-do-bind")  wrongMonadBind :: LHsExpr Id -> Type -> SDoc -wrongMonadBind rhs returning_ty -  = ptext (sLit "A do-notation statement discarded a result of type") <+> ppr returning_ty <> dot $$ +wrongMonadBind rhs elt_ty +  = ptext (sLit "A do-notation statement discarded a result of type") <+> ppr elt_ty <> dot $$      ptext (sLit "Suppress this warning by saying \"_ <- ") <> ppr rhs <> ptext (sLit "\",") $$      ptext (sLit "or by using the flag -fno-warn-wrong-do-bind")  \end{code} diff --git a/compiler/deSugar/DsGRHSs.lhs b/compiler/deSugar/DsGRHSs.lhs index a7260e2af8..d3fcf76d1c 100644 --- a/compiler/deSugar/DsGRHSs.lhs +++ b/compiler/deSugar/DsGRHSs.lhs @@ -106,11 +106,11 @@ matchGuards [] _ rhs _  	-- NB:	The success of this clause depends on the typechecker not  	-- 	wrapping the 'otherwise' in empty HsTyApp or HsWrap constructors  	--	If it does, you'll get bogus overlap warnings -matchGuards (ExprStmt e _ _ : stmts) ctx rhs rhs_ty +matchGuards (ExprStmt e _ _ _ : stmts) ctx rhs rhs_ty    | Just addTicks <- isTrueLHsExpr e = do      match_result <- matchGuards stmts ctx rhs rhs_ty      return (adjustMatchResultDs addTicks match_result) -matchGuards (ExprStmt expr _ _ : stmts) ctx rhs rhs_ty = do +matchGuards (ExprStmt expr _ _ _ : stmts) ctx rhs rhs_ty = do      match_result <- matchGuards stmts ctx rhs rhs_ty      pred_expr <- dsLExpr expr      return (mkGuardedMatchResult pred_expr match_result) diff --git a/compiler/deSugar/DsListComp.lhs b/compiler/deSugar/DsListComp.lhs index cd22b8ff8c..aabd6b0d0d 100644 --- a/compiler/deSugar/DsListComp.lhs +++ b/compiler/deSugar/DsListComp.lhs @@ -3,9 +3,10 @@  % (c) The GRASP/AQUA Project, Glasgow University, 1992-1998  % -Desugaring list comprehensions and array comprehensions +Desugaring list comprehensions, monad comprehensions and array comprehensions  \begin{code} +{-# LANGUAGE NamedFieldPuns #-}  {-# OPTIONS -fno-warn-incomplete-patterns #-}  -- The above warning supression flag is a temporary kludge.  -- While working on this module you are encouraged to remove it and fix @@ -13,11 +14,11 @@ Desugaring list comprehensions and array comprehensions  --     http://hackage.haskell.org/trac/ghc/wiki/Commentary/CodingStyle#Warnings  -- for details -module DsListComp ( dsListComp, dsPArrComp ) where +module DsListComp ( dsListComp, dsPArrComp, dsMonadComp ) where  #include "HsVersions.h" -import {-# SOURCE #-} DsExpr ( dsLExpr, dsLocalBinds ) +import {-# SOURCE #-} DsExpr ( dsExpr, dsLExpr, dsLocalBinds )  import HsSyn  import TcHsSyn @@ -37,6 +38,7 @@ import PrelNames  import SrcLoc  import Outputable  import FastString +import TcType  \end{code}  List comprehensions may be desugared in one of two ways: ``ordinary'' @@ -47,12 +49,14 @@ There will be at least one ``qualifier'' in the input.  \begin{code}  dsListComp :: [LStmt Id]  -	   -> LHsExpr Id -	   -> Type		-- Type of list elements +	   -> Type		-- Type of entire list   	   -> DsM CoreExpr -dsListComp lquals body elt_ty = do  +dsListComp lquals res_ty = do       dflags <- getDOptsDs      let quals = map unLoc lquals +        elt_ty = case tcTyConAppArgs res_ty of +                   [elt_ty] -> elt_ty +                   _ -> pprPanic "dsListComp" (ppr res_ty $$ ppr lquals)      if not (dopt Opt_EnableRewriteRules dflags) || dopt Opt_IgnoreInterfacePragmas dflags         -- Either rules are switched off, or we are ignoring what there are; @@ -60,8 +64,8 @@ dsListComp lquals body elt_ty = do         -- Wadler-style desugaring         || isParallelComp quals         -- Foldr-style desugaring can't handle parallel list comprehensions -        then deListComp quals body (mkNilExpr elt_ty) -        else mkBuildExpr elt_ty (\(c, _) (n, _) -> dfListComp c n quals body)  +        then deListComp quals (mkNilExpr elt_ty) +        else mkBuildExpr elt_ty (\(c, _) (n, _) -> dfListComp c n quals)                -- Foldr/build should be enabled, so desugar                -- into foldrs and builds @@ -72,92 +76,69 @@ dsListComp lquals body elt_ty = do      -- mix of possibly a single element in length, so we do this to leave the possibility open      isParallelComp = any isParallelStmt -    isParallelStmt (ParStmt _) = True -    isParallelStmt _           = False +    isParallelStmt (ParStmt _ _ _ _) = True +    isParallelStmt _                 = False  -- This function lets you desugar a inner list comprehension and a list of the binders  -- of that comprehension that we need in the outer comprehension into such an expression  -- and the type of the elements that it outputs (tuples of binders)  dsInnerListComp :: ([LStmt Id], [Id]) -> DsM (CoreExpr, Type) -dsInnerListComp (stmts, bndrs) = do -        expr <- dsListComp stmts (mkBigLHsVarTup bndrs) bndrs_tuple_type -        return (expr, bndrs_tuple_type) -    where -        bndrs_types = map idType bndrs -        bndrs_tuple_type = mkBigCoreTupTy bndrs_types -         +dsInnerListComp (stmts, bndrs) +  = do { expr <- dsListComp (stmts ++ [noLoc $ mkLastStmt (mkBigLHsVarTup bndrs)])  +                            (mkListTy bndrs_tuple_type) +       ; return (expr, bndrs_tuple_type) } +  where +    bndrs_tuple_type = mkBigCoreVarTupTy bndrs --- This function factors out commonality between the desugaring strategies for TransformStmt. --- Given such a statement it gives you back an expression representing how to compute the transformed --- list and the tuple that you need to bind from that list in order to proceed with your desugaring -dsTransformStmt :: Stmt Id -> DsM (CoreExpr, LPat Id) -dsTransformStmt (TransformStmt stmts binders usingExpr maybeByExpr) - = do { (expr, binders_tuple_type) <- dsInnerListComp (stmts, binders) -      ; usingExpr' <- dsLExpr usingExpr -     -      ; using_args <- -          case maybeByExpr of -            Nothing -> return [expr] -            Just byExpr -> do -                byExpr' <- dsLExpr byExpr -                 -                us <- newUniqueSupply -                [tuple_binder] <- newSysLocalsDs [binders_tuple_type] -                let byExprWrapper = mkTupleCase us binders byExpr' tuple_binder (Var tuple_binder) -                 -                return [Lam tuple_binder byExprWrapper, expr] - -      ; let inner_list_expr = mkApps usingExpr' ((Type binders_tuple_type) : using_args) -            pat = mkBigLHsVarPatTup binders -      ; return (inner_list_expr, pat) } -      -- This function factors out commonality between the desugaring strategies for GroupStmt.  -- Given such a statement it gives you back an expression representing how to compute the transformed  -- list and the tuple that you need to bind from that list in order to proceed with your desugaring -dsGroupStmt :: Stmt Id -> DsM (CoreExpr, LPat Id) -dsGroupStmt (GroupStmt stmts binderMap by using) = do -    let (fromBinders, toBinders) = unzip binderMap -         -        fromBindersTypes = map idType fromBinders -        toBindersTypes = map idType toBinders -         -        toBindersTupleType = mkBigCoreTupTy toBindersTypes +dsTransStmt :: Stmt Id -> DsM (CoreExpr, LPat Id) +dsTransStmt (TransStmt { trS_form = form, trS_stmts = stmts, trS_bndrs = binderMap +                       , trS_by = by, trS_using = using }) = do +    let (from_bndrs, to_bndrs) = unzip binderMap +        from_bndrs_tys  = map idType from_bndrs +        to_bndrs_tys    = map idType to_bndrs +        to_bndrs_tup_ty = mkBigCoreTupTy to_bndrs_tys      -- Desugar an inner comprehension which outputs a list of tuples of the "from" binders -    (expr, from_tup_ty) <- dsInnerListComp (stmts, fromBinders) +    (expr, from_tup_ty) <- dsInnerListComp (stmts, from_bndrs)      -- Work out what arguments should be supplied to that expression: i.e. is an extraction      -- function required? If so, create that desugared function and add to arguments -    usingExpr' <- dsLExpr (either id noLoc using) +    usingExpr' <- dsLExpr using      usingArgs <- case by of                     Nothing   -> return [expr]   		   Just by_e -> do { by_e' <- dsLExpr by_e -                                   ; us <- newUniqueSupply -                                   ; [from_tup_id] <- newSysLocalsDs [from_tup_ty] -                                   ; let by_wrap = mkTupleCase us fromBinders by_e'  -                                                   from_tup_id (Var from_tup_id) -                                   ; return [Lam from_tup_id by_wrap, expr] } +                                   ; lam <- matchTuple from_bndrs by_e' +                                   ; return [lam, expr] }      -- Create an unzip function for the appropriate arity and element types and find "map" -    (unzip_fn, unzip_rhs) <- mkUnzipBind fromBindersTypes +    unzip_stuff <- mkUnzipBind form from_bndrs_tys      map_id <- dsLookupGlobalId mapName      -- Generate the expressions to build the grouped list      let -- First we apply the grouping function to the inner list -        inner_list_expr = mkApps usingExpr' ((Type from_tup_ty) : usingArgs) +        inner_list_expr = mkApps usingExpr' usingArgs          -- Then we map our "unzip" across it to turn the lists of tuples into tuples of lists          -- We make sure we instantiate the type variable "a" to be a list of "from" tuples and          -- the "b" to be a tuple of "to" lists! -        unzipped_inner_list_expr = mkApps (Var map_id)  -            [Type (mkListTy from_tup_ty), Type toBindersTupleType, Var unzip_fn, inner_list_expr]          -- Then finally we bind the unzip function around that expression -        bound_unzipped_inner_list_expr = Let (Rec [(unzip_fn, unzip_rhs)]) unzipped_inner_list_expr -     -    -- Build a pattern that ensures the consumer binds into the NEW binders, which hold lists rather than single values -    let pat = mkBigLHsVarPatTup toBinders +        bound_unzipped_inner_list_expr  +          = case unzip_stuff of +              Nothing -> inner_list_expr +              Just (unzip_fn, unzip_rhs) -> Let (Rec [(unzip_fn, unzip_rhs)]) $ +                                            mkApps (Var map_id) $ +                                            [ Type (mkListTy from_tup_ty) +                                            , Type to_bndrs_tup_ty +                                            , Var unzip_fn +                                            , inner_list_expr] + +    -- Build a pattern that ensures the consumer binds into the NEW binders,  +    -- which hold lists rather than single values +    let pat = mkBigLHsVarPatTup to_bndrs      return (bound_unzipped_inner_list_expr, pat) -      \end{code}  %************************************************************************ @@ -226,53 +207,50 @@ with the Unboxed variety.  \begin{code} -deListComp :: [Stmt Id] -> LHsExpr Id -> CoreExpr -> DsM CoreExpr - -deListComp (ParStmt stmtss_w_bndrs : quals) body list -  = do -    exps_and_qual_tys <- mapM dsInnerListComp stmtss_w_bndrs -    let (exps, qual_tys) = unzip exps_and_qual_tys -     -    (zip_fn, zip_rhs) <- mkZipBind qual_tys +deListComp :: [Stmt Id] -> CoreExpr -> DsM CoreExpr -	-- Deal with [e | pat <- zip l1 .. ln] in example above -    deBindComp pat (Let (Rec [(zip_fn, zip_rhs)]) (mkApps (Var zip_fn) exps))  -		   quals body list +deListComp [] _ = panic "deListComp" -  where  -	bndrs_s = map snd stmtss_w_bndrs - -	-- pat is the pattern ((x1,..,xn), (y1,..,ym)) in the example above -	pat  = mkBigLHsPatTup pats -	pats = map mkBigLHsVarPatTup bndrs_s - -	-- Last: the one to return -deListComp [] body list = do    -- Figure 7.4, SLPJ, p 135, rule C above -    core_body <- dsLExpr body -    return (mkConsExpr (exprType core_body) core_body list) +deListComp (LastStmt body _ : quals) list  +  =     -- Figure 7.4, SLPJ, p 135, rule C above +    ASSERT( null quals ) +    do { core_body <- dsLExpr body +       ; return (mkConsExpr (exprType core_body) core_body list) }  	-- Non-last: must be a guard -deListComp (ExprStmt guard _ _ : quals) body list = do  -- rule B above +deListComp (ExprStmt guard _ _ _ : quals) list = do  -- rule B above      core_guard <- dsLExpr guard -    core_rest <- deListComp quals body list +    core_rest <- deListComp quals list      return (mkIfThenElse core_guard core_rest list)  -- [e | let B, qs] = let B in [e | qs] -deListComp (LetStmt binds : quals) body list = do -    core_rest <- deListComp quals body list +deListComp (LetStmt binds : quals) list = do +    core_rest <- deListComp quals list      dsLocalBinds binds core_rest -deListComp (stmt@(TransformStmt {}) : quals) body list = do -    (inner_list_expr, pat) <- dsTransformStmt stmt -    deBindComp pat inner_list_expr quals body list +deListComp (stmt@(TransStmt {}) : quals) list = do +    (inner_list_expr, pat) <- dsTransStmt stmt +    deBindComp pat inner_list_expr quals list -deListComp (stmt@(GroupStmt {}) : quals) body list = do -    (inner_list_expr, pat) <- dsGroupStmt stmt -    deBindComp pat inner_list_expr quals body list - -deListComp (BindStmt pat list1 _ _ : quals) body core_list2 = do -- rule A' above +deListComp (BindStmt pat list1 _ _ : quals) core_list2 = do -- rule A' above      core_list1 <- dsLExpr list1 -    deBindComp pat core_list1 quals body core_list2 +    deBindComp pat core_list1 quals core_list2 + +deListComp (ParStmt stmtss_w_bndrs _ _ _ : quals) list +  = do { exps_and_qual_tys <- mapM dsInnerListComp stmtss_w_bndrs +       ; let (exps, qual_tys) = unzip exps_and_qual_tys +     +       ; (zip_fn, zip_rhs) <- mkZipBind qual_tys + +	-- Deal with [e | pat <- zip l1 .. ln] in example above +       ; deBindComp pat (Let (Rec [(zip_fn, zip_rhs)]) (mkApps (Var zip_fn) exps))  +		    quals list } +  where  +	bndrs_s = map snd stmtss_w_bndrs + +	-- pat is the pattern ((x1,..,xn), (y1,..,ym)) in the example above +	pat  = mkBigLHsPatTup pats +	pats = map mkBigLHsVarPatTup bndrs_s  \end{code} @@ -280,10 +258,9 @@ deListComp (BindStmt pat list1 _ _ : quals) body core_list2 = do -- rule A' abov  deBindComp :: OutPat Id             -> CoreExpr             -> [Stmt Id] -           -> LHsExpr Id             -> CoreExpr             -> DsM (Expr Id) -deBindComp pat core_list1 quals body core_list2 = do +deBindComp pat core_list1 quals core_list2 = do      let          u3_ty@u1_ty = exprType core_list1	-- two names, same thing @@ -300,7 +277,7 @@ deBindComp pat core_list1 quals body core_list2 = do          core_fail   = App (Var h) (Var u3)          letrec_body = App (Var h) core_list1 -    rest_expr <- deListComp quals body core_fail +    rest_expr <- deListComp quals core_fail      core_match <- matchSimply (Var u2) (StmtCtxt ListComp) pat rest_expr core_fail	      let @@ -335,48 +312,43 @@ TE[ e | p <- l , q ] c n = let  \begin{code}  dfListComp :: Id -> Id -- 'c' and 'n'          -> [Stmt Id]   -- the rest of the qual's -        -> LHsExpr Id          -> DsM CoreExpr -	-- Last: the one to return -dfListComp c_id n_id [] body = do -    core_body <- dsLExpr body -    return (mkApps (Var c_id) [core_body, Var n_id]) +dfListComp _ _ [] = panic "dfListComp" + +dfListComp c_id n_id (LastStmt body _ : quals)  +  = ASSERT( null quals ) +    do { core_body <- dsLExpr body +       ; return (mkApps (Var c_id) [core_body, Var n_id]) }  	-- Non-last: must be a guard -dfListComp c_id n_id (ExprStmt guard _ _  : quals) body = do +dfListComp c_id n_id (ExprStmt guard _ _ _  : quals) = do      core_guard <- dsLExpr guard -    core_rest <- dfListComp c_id n_id quals body +    core_rest <- dfListComp c_id n_id quals      return (mkIfThenElse core_guard core_rest (Var n_id)) -dfListComp c_id n_id (LetStmt binds : quals) body = do +dfListComp c_id n_id (LetStmt binds : quals) = do      -- new in 1.3, local bindings -    core_rest <- dfListComp c_id n_id quals body +    core_rest <- dfListComp c_id n_id quals      dsLocalBinds binds core_rest -dfListComp c_id n_id (stmt@(TransformStmt {}) : quals) body = do -    (inner_list_expr, pat) <- dsTransformStmt stmt -    -- Anyway, we bind the newly transformed list via the generic binding function -    dfBindComp c_id n_id (pat, inner_list_expr) quals body - -dfListComp c_id n_id (stmt@(GroupStmt {}) : quals) body = do -    (inner_list_expr, pat) <- dsGroupStmt stmt +dfListComp c_id n_id (stmt@(TransStmt {}) : quals) = do +    (inner_list_expr, pat) <- dsTransStmt stmt      -- Anyway, we bind the newly grouped list via the generic binding function -    dfBindComp c_id n_id (pat, inner_list_expr) quals body +    dfBindComp c_id n_id (pat, inner_list_expr) quals  -dfListComp c_id n_id (BindStmt pat list1 _ _ : quals) body = do +dfListComp c_id n_id (BindStmt pat list1 _ _ : quals) = do      -- evaluate the two lists      core_list1 <- dsLExpr list1      -- Do the rest of the work in the generic binding builder -    dfBindComp c_id n_id (pat, core_list1) quals body +    dfBindComp c_id n_id (pat, core_list1) quals  dfBindComp :: Id -> Id	        -- 'c' and 'n'         -> (LPat Id, CoreExpr)  	   -> [Stmt Id] 	        -- the rest of the qual's -	   -> LHsExpr Id  	   -> DsM CoreExpr -dfBindComp c_id n_id (pat, core_list1) quals body = do +dfBindComp c_id n_id (pat, core_list1) quals = do      -- find the required type      let x_ty   = hsLPatType pat          b_ty   = idType n_id @@ -385,7 +357,7 @@ dfBindComp c_id n_id (pat, core_list1) quals body = do      [b, x] <- newSysLocalsDs [b_ty, x_ty]      -- build rest of the comprehesion -    core_rest <- dfListComp c_id b quals body +    core_rest <- dfListComp c_id b quals      -- build the pattern match      core_expr <- matchSimply (Var x) (StmtCtxt ListComp) @@ -439,7 +411,7 @@ mkZipBind elt_tys = do  			-- Increasing order of tag -mkUnzipBind :: [Type] -> DsM (Id, CoreExpr) +mkUnzipBind :: TransForm -> [Type] -> DsM (Maybe (Id, CoreExpr))  -- mkUnzipBind [t1, t2]   -- = (unzip, \ys :: [(t1, t2)] -> foldr (\ax :: (t1, t2) axs :: ([t1], [t2])  --     -> case ax of @@ -449,28 +421,29 @@ mkUnzipBind :: [Type] -> DsM (Id, CoreExpr)  --      ys)  --   -- We use foldr here in all cases, even if rules are turned off, because we may as well! -mkUnzipBind elt_tys = do -    ax  <- newSysLocalDs elt_tuple_ty -    axs <- newSysLocalDs elt_list_tuple_ty -    ys  <- newSysLocalDs elt_tuple_list_ty -    xs  <- mapM newSysLocalDs elt_tys -    xss <- mapM newSysLocalDs elt_list_tys +mkUnzipBind ThenForm _ + = return Nothing    -- No unzipping for ThenForm +mkUnzipBind _ elt_tys  +  = do { ax  <- newSysLocalDs elt_tuple_ty +       ; axs <- newSysLocalDs elt_list_tuple_ty +       ; ys  <- newSysLocalDs elt_tuple_list_ty +       ; xs  <- mapM newSysLocalDs elt_tys +       ; xss <- mapM newSysLocalDs elt_list_tys -    unzip_fn <- newSysLocalDs unzip_fn_ty - -    [us1, us2] <- sequence [newUniqueSupply, newUniqueSupply] - -    let nil_tuple = mkBigCoreTup (map mkNilExpr elt_tys) -         -        concat_expressions = map mkConcatExpression (zip3 elt_tys (map Var xs) (map Var xss)) -        tupled_concat_expression = mkBigCoreTup concat_expressions -         -        folder_body_inner_case = mkTupleCase us1 xss tupled_concat_expression axs (Var axs) -        folder_body_outer_case = mkTupleCase us2 xs folder_body_inner_case ax (Var ax) -        folder_body = mkLams [ax, axs] folder_body_outer_case -         -    unzip_body <- mkFoldrExpr elt_tuple_ty elt_list_tuple_ty folder_body nil_tuple (Var ys) -    return (unzip_fn, mkLams [ys] unzip_body) +       ; unzip_fn <- newSysLocalDs unzip_fn_ty + +       ; [us1, us2] <- sequence [newUniqueSupply, newUniqueSupply] + +       ; let nil_tuple = mkBigCoreTup (map mkNilExpr elt_tys) +    	     concat_expressions = map mkConcatExpression (zip3 elt_tys (map Var xs) (map Var xss)) +    	     tupled_concat_expression = mkBigCoreTup concat_expressions +    	     +    	     folder_body_inner_case = mkTupleCase us1 xss tupled_concat_expression axs (Var axs) +    	     folder_body_outer_case = mkTupleCase us2 xs folder_body_inner_case ax (Var ax) +    	     folder_body = mkLams [ax, axs] folder_body_outer_case +    	     +       ; unzip_body <- mkFoldrExpr elt_tuple_ty elt_list_tuple_ty folder_body nil_tuple (Var ys) +       ; return (Just (unzip_fn, mkLams [ys] unzip_body)) }    where      elt_tuple_ty       = mkBigCoreTupTy elt_tys      elt_tuple_list_ty  = mkListTy elt_tuple_ty @@ -480,9 +453,6 @@ mkUnzipBind elt_tys = do      unzip_fn_ty        = elt_tuple_list_ty `mkFunTy` elt_list_tuple_ty      mkConcatExpression (list_element_ty, head, tail) = mkConsExpr list_element_ty head tail -             -             -  \end{code}  %************************************************************************ @@ -498,11 +468,10 @@ mkUnzipBind elt_tys = do  --   [:e | qss:] = <<[:e | qss:]>> () [:():]  --  dsPArrComp :: [Stmt Id]  -            -> LHsExpr Id -            -> Type		    -- Don't use; called with `undefined' below              -> DsM CoreExpr -dsPArrComp [ParStmt qss] body _  =  -- parallel comprehension -  dePArrParComp qss body + +-- Special case for parallel comprehension +dsPArrComp (ParStmt qss _ _ _ : quals) = dePArrParComp qss quals  -- Special case for simple generators:  -- @@ -513,7 +482,7 @@ dsPArrComp [ParStmt qss] body _  =  -- parallel comprehension  --  <<[:e' | p <- e, qs:]>> =   --    <<[:e' | qs:]>> p (filterP (\x -> case x of {p -> True; _ -> False}) e)  -- -dsPArrComp (BindStmt p e _ _ : qs) body _ = do +dsPArrComp (BindStmt p e _ _ : qs) = do      filterP <- dsLookupDPHId filterPName      ce <- dsLExpr e      let ety'ce  = parrElemType ce @@ -523,38 +492,41 @@ dsPArrComp (BindStmt p e _ _ : qs) body _ = do      pred <- matchSimply (Var v) (StmtCtxt PArrComp) p true false      let gen | isIrrefutableHsPat p = ce              | otherwise            = mkApps (Var filterP) [Type ety'ce, mkLams [v] pred, ce] -    dePArrComp qs body p gen +    dePArrComp qs p gen -dsPArrComp qs            body _  = do -- no ParStmt in `qs' +dsPArrComp qs = do -- no ParStmt in `qs'      sglP <- dsLookupDPHId singletonPName      let unitArray = mkApps (Var sglP) [Type unitTy, mkCoreTup []] -    dePArrComp qs body (noLoc $ WildPat unitTy) unitArray +    dePArrComp qs (noLoc $ WildPat unitTy) unitArray  -- the work horse  --  dePArrComp :: [Stmt Id]  -	   -> LHsExpr Id  	   -> LPat Id		-- the current generator pattern  	   -> CoreExpr		-- the current generator expression  	   -> DsM CoreExpr + +dePArrComp [] _ _ = panic "dePArrComp" +  --  --  <<[:e' | :]>> pa ea = mapP (\pa -> e') ea  -- -dePArrComp [] e' pa cea = do -    mapP <- dsLookupDPHId mapPName -    let ty = parrElemType cea -    (clam, ty'e') <- deLambda ty pa e' -    return $ mkApps (Var mapP) [Type ty, Type ty'e', clam, cea] +dePArrComp (LastStmt e' _ : quals) pa cea +  = ASSERT( null quals ) +    do { mapP <- dsLookupDPHId mapPName +       ; let ty = parrElemType cea +       ; (clam, ty'e') <- deLambda ty pa e' +       ; return $ mkApps (Var mapP) [Type ty, Type ty'e', clam, cea] }  --  --  <<[:e' | b, qs:]>> pa ea = <<[:e' | qs:]>> pa (filterP (\pa -> b) ea)  -- -dePArrComp (ExprStmt b _ _ : qs) body pa cea = do +dePArrComp (ExprStmt b _ _ _ : qs) pa cea = do      filterP <- dsLookupDPHId filterPName      let ty = parrElemType cea      (clam,_) <- deLambda ty pa b -    dePArrComp qs body pa (mkApps (Var filterP) [Type ty, clam, cea]) +    dePArrComp qs pa (mkApps (Var filterP) [Type ty, clam, cea])  --  --  <<[:e' | p <- e, qs:]>> pa ea = @@ -569,7 +541,7 @@ dePArrComp (ExprStmt b _ _ : qs) body pa cea = do  --    in  --    <<[:e' | qs:]>> (pa, p) (crossMapP ea ef)  -- -dePArrComp (BindStmt p e _ _ : qs) body pa cea = do +dePArrComp (BindStmt p e _ _ : qs) pa cea = do      filterP <- dsLookupDPHId filterPName      crossMapP <- dsLookupDPHId crossMapPName      ce <- dsLExpr e @@ -585,7 +557,7 @@ dePArrComp (BindStmt p e _ _ : qs) body pa cea = do      let ety'cef = ety'ce		    -- filter doesn't change the element type          pa'     = mkLHsPatTup [pa, p] -    dePArrComp qs body pa' (mkApps (Var crossMapP)  +    dePArrComp qs pa' (mkApps (Var crossMapP)                                    [Type ety'cea, Type ety'cef, cea, clam])  --  --  <<[:e' | let ds, qs:]>> pa ea =  @@ -594,7 +566,7 @@ dePArrComp (BindStmt p e _ _ : qs) body pa cea = do  --  where  --    {x_1, ..., x_n} = DV (ds)		-- Defined Variables  -- -dePArrComp (LetStmt ds : qs) body pa cea = do +dePArrComp (LetStmt ds : qs) pa cea = do      mapP <- dsLookupDPHId mapPName      let xs     = collectLocalBinders ds          ty'cea = parrElemType cea @@ -609,14 +581,14 @@ dePArrComp (LetStmt ds : qs) body pa cea = do      ccase <- matchSimply (Var v) (StmtCtxt PArrComp) pa projBody cerr      let pa'    = mkLHsPatTup [pa, mkLHsPatTup (map nlVarPat xs)]          proj   = mkLams [v] ccase -    dePArrComp qs body pa' (mkApps (Var mapP)  +    dePArrComp qs pa' (mkApps (Var mapP)                                      [Type ty'cea, Type errTy, proj, cea])  --  -- The parser guarantees that parallel comprehensions can only appear as  -- singeltons qualifier lists, which we already special case in the caller.  -- So, encountering one here is a bug.  -- -dePArrComp (ParStmt _ : _) _ _ _ =  +dePArrComp (ParStmt _ _ _ _ : _) _ _ =     panic "DsListComp.dePArrComp: malformed comprehension AST"  --  <<[:e' | qs | qss:]>> pa ea =  @@ -625,17 +597,17 @@ dePArrComp (ParStmt _ : _) _ _ _ =  --    where  --      {x_1, ..., x_n} = DV (qs)  -- -dePArrParComp :: [([LStmt Id], [Id])] -> LHsExpr Id -> DsM CoreExpr -dePArrParComp qss body = do +dePArrParComp :: [([LStmt Id], [Id])] -> [Stmt Id] -> DsM CoreExpr +dePArrParComp qss quals = do      (pQss, ceQss) <- deParStmt qss -    dePArrComp [] body pQss ceQss +    dePArrComp quals pQss ceQss    where      deParStmt []             =        -- empty parallel statement lists have no source representation        panic "DsListComp.dePArrComp: Empty parallel list comprehension"      deParStmt ((qs, xs):qss) = do        -- first statement        let res_expr = mkLHsVarTuple xs -      cqs <- dsPArrComp (map unLoc qs) res_expr undefined +      cqs <- dsPArrComp (map unLoc qs ++ [mkLastStmt res_expr])        parStmts qss (mkLHsVarPatTup xs) cqs      ---      parStmts []             pa cea = return (pa, cea) @@ -644,7 +616,7 @@ dePArrParComp qss body = do        let pa'      = mkLHsPatTup [pa, mkLHsVarPatTup xs]            ty'cea   = parrElemType cea            res_expr = mkLHsVarTuple xs -      cqs <- dsPArrComp (map unLoc qs) res_expr undefined +      cqs <- dsPArrComp (map unLoc qs ++ [mkLastStmt res_expr])        let ty'cqs = parrElemType cqs            cea'   = mkApps (Var zipP) [Type ty'cea, Type ty'cqs, cea, cqs]        parStmts qss pa' cea' @@ -682,3 +654,222 @@ parrElemType e  =      _							  -> panic        "DsListComp.parrElemType: not a parallel array type"  \end{code} + +Translation for monad comprehensions + +\begin{code} +-- Entry point for monad comprehension desugaring +dsMonadComp :: [LStmt Id] -> DsM CoreExpr +dsMonadComp stmts = dsMcStmts stmts + +dsMcStmts :: [LStmt Id] -> DsM CoreExpr +dsMcStmts []                    = panic "dsMcStmts" +dsMcStmts (L loc stmt : lstmts) = putSrcSpanDs loc (dsMcStmt stmt lstmts) + +--------------- +dsMcStmt :: Stmt Id -> [LStmt Id] -> DsM CoreExpr + +dsMcStmt (LastStmt body ret_op) stmts +  = ASSERT( null stmts ) +    do { body' <- dsLExpr body +       ; ret_op' <- dsExpr ret_op +       ; return (App ret_op' body') } + +--   [ .. | let binds, stmts ] +dsMcStmt (LetStmt binds) stmts  +  = do { rest <- dsMcStmts stmts +       ; dsLocalBinds binds rest } + +--   [ .. | a <- m, stmts ] +dsMcStmt (BindStmt pat rhs bind_op fail_op) stmts +  = do { rhs' <- dsLExpr rhs +       ; dsMcBindStmt pat rhs' bind_op fail_op stmts } + +-- Apply `guard` to the `exp` expression +-- +--   [ .. | exp, stmts ] +-- +dsMcStmt (ExprStmt exp then_exp guard_exp _) stmts  +  = do { exp'       <- dsLExpr exp +       ; guard_exp' <- dsExpr guard_exp +       ; then_exp'  <- dsExpr then_exp +       ; rest       <- dsMcStmts stmts +       ; return $ mkApps then_exp' [ mkApps guard_exp' [exp'] +                                   , rest ] } + +-- Group statements desugar like this: +-- +--   [| (q, then group by e using f); rest |] +--   --->  f {qt} (\qv -> e) [| q; return qv |] >>= \ n_tup ->  +--         case unzip n_tup of qv' -> [| rest |] +-- +-- where   variables (v1:t1, ..., vk:tk) are bound by q +--         qv = (v1, ..., vk) +--         qt = (t1, ..., tk) +--         (>>=) :: m2 a -> (a -> m3 b) -> m3 b +--         f :: forall a. (a -> t) -> m1 a -> m2 (n a) +--         n_tup :: n qt +--         unzip :: n qt -> (n t1, ..., n tk)    (needs Functor n) + +dsMcStmt (TransStmt { trS_stmts = stmts, trS_bndrs = bndrs +                    , trS_by = by, trS_using = using +                    , trS_ret = return_op, trS_bind = bind_op +                    , trS_fmap = fmap_op, trS_form = form }) stmts_rest +  = do { let (from_bndrs, to_bndrs) = unzip bndrs +             from_bndr_tys          = map idType from_bndrs	-- Types ty + +       -- Desugar an inner comprehension which outputs a list of tuples of the "from" binders +       ; expr <- dsInnerMonadComp stmts from_bndrs return_op + +       -- Work out what arguments should be supplied to that expression: i.e. is an extraction +       -- function required? If so, create that desugared function and add to arguments +       ; usingExpr' <- dsLExpr using +       ; usingArgs <- case by of +                        Nothing   -> return [expr] +                        Just by_e -> do { by_e' <- dsLExpr by_e +                                        ; lam <- matchTuple from_bndrs by_e' +                                        ; return [lam, expr] } + +       -- Generate the expressions to build the grouped list +       -- Build a pattern that ensures the consumer binds into the NEW binders,  +       -- which hold monads rather than single values +       ; bind_op' <- dsExpr bind_op +       ; let bind_ty  = exprType bind_op'    -- m2 (n (a,b,c)) -> (n (a,b,c) -> r1) -> r2 +             n_tup_ty = funArgTy $ funArgTy $ funResultTy bind_ty   -- n (a,b,c) +             tup_n_ty = mkBigCoreVarTupTy to_bndrs + +       ; body       <- dsMcStmts stmts_rest +       ; n_tup_var  <- newSysLocalDs n_tup_ty +       ; tup_n_var  <- newSysLocalDs tup_n_ty +       ; tup_n_expr <- mkMcUnzipM form fmap_op n_tup_var from_bndr_tys +       ; us         <- newUniqueSupply +       ; let rhs'  = mkApps usingExpr' usingArgs +             body' = mkTupleCase us to_bndrs body tup_n_var tup_n_expr +		    +       ; return (mkApps bind_op' [rhs', Lam n_tup_var body']) } + +-- Parallel statements. Use `Control.Monad.Zip.mzip` to zip parallel +-- statements, for example: +-- +--   [ body | qs1 | qs2 | qs3 ] +--     ->  [ body | (bndrs1, (bndrs2, bndrs3))  +--                     <- [bndrs1 | qs1] `mzip` ([bndrs2 | qs2] `mzip` [bndrs3 | qs3]) ] +-- +-- where `mzip` has type +--   mzip :: forall a b. m a -> m b -> m (a,b) +-- NB: we need a polymorphic mzip because we call it several times + +dsMcStmt (ParStmt pairs mzip_op bind_op return_op) stmts_rest + = do  { exps_w_tys  <- mapM ds_inner pairs   -- Pairs (exp :: m ty, ty) +       ; mzip_op'    <- dsExpr mzip_op + +       ; let -- The pattern variables +             pats = map (mkBigLHsVarPatTup . snd) pairs +             -- Pattern with tuples of variables +             -- [v1,v2,v3]  =>  (v1, (v2, v3)) +             pat = foldr1 (\p1 p2 -> mkLHsPatTup [p1, p2]) pats +	     (rhs, _) = foldr1 (\(e1,t1) (e2,t2) ->  +                                 (mkApps mzip_op' [Type t1, Type t2, e1, e2], +                                  mkBoxedTupleTy [t1,t2]))  +                               exps_w_tys + +       ; dsMcBindStmt pat rhs bind_op noSyntaxExpr stmts_rest } +  where +    ds_inner (stmts, bndrs) = do { exp <- dsInnerMonadComp stmts bndrs mono_ret_op +                                 ; return (exp, tup_ty) } +       where  +         mono_ret_op = HsWrap (WpTyApp tup_ty) return_op +         tup_ty      = mkBigCoreVarTupTy bndrs + +dsMcStmt stmt _ = pprPanic "dsMcStmt: unexpected stmt" (ppr stmt) + + +matchTuple :: [Id] -> CoreExpr -> DsM CoreExpr +-- (matchTuple [a,b,c] body) +--       returns the Core term +--  \x. case x of (a,b,c) -> body  +matchTuple ids body +  = do { us <- newUniqueSupply +       ; tup_id <- newSysLocalDs (mkBigCoreVarTupTy ids) +       ; return (Lam tup_id $ mkTupleCase us ids body tup_id (Var tup_id)) } + +-- general `rhs' >>= \pat -> stmts` desugaring where `rhs'` is already a +-- desugared `CoreExpr` +dsMcBindStmt :: LPat Id +             -> CoreExpr        -- ^ the desugared rhs of the bind statement +             -> SyntaxExpr Id +             -> SyntaxExpr Id +             -> [LStmt Id] +             -> DsM CoreExpr +dsMcBindStmt pat rhs' bind_op fail_op stmts +  = do  { body     <- dsMcStmts stmts  +        ; bind_op' <- dsExpr bind_op +        ; var      <- selectSimpleMatchVarL pat +        ; let bind_ty = exprType bind_op' 	-- rhs -> (pat -> res1) -> res2 +              res1_ty = funResultTy (funArgTy (funResultTy bind_ty)) +        ; match <- matchSinglePat (Var var) (StmtCtxt DoExpr) pat +                                  res1_ty (cantFailMatchResult body) +        ; match_code <- handle_failure pat match fail_op +        ; return (mkApps bind_op' [rhs', Lam var match_code]) } + +  where +    -- In a monad comprehension expression, pattern-match failure just calls +    -- the monadic `fail` rather than throwing an exception +    handle_failure pat match fail_op +      | matchCanFail match +        = do { fail_op' <- dsExpr fail_op +             ; fail_msg <- mkStringExpr (mk_fail_msg pat) +             ; extractMatchResult match (App fail_op' fail_msg) } +      | otherwise +        = extractMatchResult match (error "It can't fail")  + +    mk_fail_msg :: Located e -> String +    mk_fail_msg pat = "Pattern match failure in monad comprehension at " ++  +                      showSDoc (ppr (getLoc pat)) + +-- Desugar nested monad comprehensions, for example in `then..` constructs +--    dsInnerMonadComp quals [a,b,c] ret_op +-- returns the desugaring of  +--       [ (a,b,c) | quals ] + +dsInnerMonadComp :: [LStmt Id] +                 -> [Id]	-- Return a tuple of these variables +                 -> HsExpr Id	-- The monomorphic "return" operator +                 -> DsM CoreExpr +dsInnerMonadComp stmts bndrs ret_op +  = dsMcStmts (stmts ++ [noLoc (LastStmt (mkBigLHsVarTup bndrs) ret_op)]) + +-- The `unzip` function for `GroupStmt` in a monad comprehensions +-- +--   unzip :: m (a,b,..) -> (m a,m b,..) +--   unzip m_tuple = ( liftM selN1 m_tuple +--                   , liftM selN2 m_tuple +--                   , .. ) +-- +--   mkMcUnzipM fmap ys [t1, t2] +--     = ( fmap (selN1 :: (t1, t2) -> t1) ys +--       , fmap (selN2 :: (t1, t2) -> t2) ys ) + +mkMcUnzipM :: TransForm +           -> SyntaxExpr TcId	-- fmap +	   -> Id		-- Of type n (a,b,c) +	   -> [Type]		-- [a,b,c] +	   -> DsM CoreExpr	-- Of type (n a, n b, n c) +mkMcUnzipM ThenForm _ ys _ 	 +  = return (Var ys) -- No unzipping to do + +mkMcUnzipM _ fmap_op ys elt_tys +  = do { fmap_op' <- dsExpr fmap_op +       ; xs       <- mapM newSysLocalDs elt_tys +       ; let tup_ty = mkBigCoreTupTy elt_tys +       ; tup_xs   <- newSysLocalDs tup_ty +  +       ; let mk_elt i = mkApps fmap_op'  -- fmap :: forall a b. (a -> b) -> n a -> n b +                           [ Type tup_ty, Type (elt_tys !! i) +                           , mk_sel i, Var ys] + +             mk_sel n = Lam tup_xs $  +                        mkTupleSelector xs (xs !! n) tup_xs (Var tup_xs) + +       ; return (mkBigCoreTup (map mk_elt [0..length elt_tys - 1])) } +\end{code} diff --git a/compiler/deSugar/DsMeta.hs b/compiler/deSugar/DsMeta.hs index e34c6960d7..e68173a59d 100644 --- a/compiler/deSugar/DsMeta.hs +++ b/compiler/deSugar/DsMeta.hs @@ -721,23 +721,19 @@ repE (HsLet bs e)         = do { (ss,ds) <- repBinds bs  			       ; wrapGenSyms ss z }  -- FIXME: I haven't got the types here right yet -repE e@(HsDo ctxt sts body _)  +repE e@(HsDo ctxt sts _)    | case ctxt of { DoExpr -> True; GhciStmt -> True; _ -> False }   = do { (ss,zs) <- repLSts sts;  -	body'	<- addBinds ss $ repLE body; -	ret	<- repNoBindSt body';	 -        e'      <- repDoE (nonEmptyCoreList (zs ++ [ret])); +        e'      <- repDoE (nonEmptyCoreList zs);          wrapGenSyms ss e' }   | ListComp <- ctxt   = do { (ss,zs) <- repLSts sts;  -	body'	<- addBinds ss $ repLE body; -	ret	<- repNoBindSt body';	 -        e'      <- repComp (nonEmptyCoreList (zs ++ [ret])); +        e'      <- repComp (nonEmptyCoreList zs);          wrapGenSyms ss e' }    | otherwise -  = notHandled "mdo and [: :]" (ppr e) +  = notHandled "mdo, monad comprehension and [: :]" (ppr e)  repE (ExplicitList _ es) = do { xs <- repLEs es; repListExp xs }  repE e@(ExplicitPArr _ _) = notHandled "Parallel arrays" (ppr e) @@ -817,7 +813,7 @@ repGuards other       wrapGenSyms (concat xs) gd }    where       process :: LGRHS Name -> DsM ([GenSymBind], (Core (TH.Q (TH.Guard, TH.Exp)))) -    process (L _ (GRHS [L _ (ExprStmt e1 _ _)] e2)) +    process (L _ (GRHS [L _ (ExprStmt e1 _ _ _)] e2))             = do { x <- repLNormalGE e1 e2;                    return ([], x) }      process (L _ (GRHS ss rhs)) @@ -876,7 +872,7 @@ repSts (LetStmt bs : ss) =        ; z <- repLetSt ds        ; (ss2,zs) <- addBinds ss1 (repSts ss)        ; return (ss1++ss2, z : zs) }  -repSts (ExprStmt e _ _ : ss) =        +repSts (ExprStmt e _ _ _ : ss) =            do { e2 <- repLE e        ; z <- repNoBindSt e2         ; (ss2,zs) <- repSts ss diff --git a/compiler/deSugar/Match.lhs b/compiler/deSugar/Match.lhs index 00a162e4df..1a044d3471 100644 --- a/compiler/deSugar/Match.lhs +++ b/compiler/deSugar/Match.lhs @@ -523,7 +523,7 @@ tidy1 _ (LitPat lit)  -- NPats: we *might* be able to replace these w/ a simpler form  tidy1 _ (NPat lit mb_neg eq) -  = return (idDsWrapper, tidyNPat lit mb_neg eq) +  = return (idDsWrapper, tidyNPat tidyLitPat lit mb_neg eq)  -- BangPatterns: Pattern matching is already strict in constructors,  -- tuples etc, so the last case strips off the bang for thoses patterns. diff --git a/compiler/deSugar/MatchLit.lhs b/compiler/deSugar/MatchLit.lhs index 5e5e81d2ba..be112e09a7 100644 --- a/compiler/deSugar/MatchLit.lhs +++ b/compiler/deSugar/MatchLit.lhs @@ -152,8 +152,14 @@ tidyLitPat (HsString s)  tidyLitPat lit = LitPat lit  ---------------- -tidyNPat :: HsOverLit Id -> Maybe (SyntaxExpr Id) -> SyntaxExpr Id -> Pat Id -tidyNPat (OverLit val False _ ty) mb_neg _ +tidyNPat :: (HsLit -> Pat Id)	-- How to tidy a LitPat +	    	 -- We need this argument because tidyNPat is called +		 -- both by Match and by Check, but they tidy LitPats  +		 -- slightly differently; and we must desugar  +		 -- literals consistently (see Trac #5117) +         -> HsOverLit Id -> Maybe (SyntaxExpr Id) -> SyntaxExpr Id  +         -> Pat Id +tidyNPat tidy_lit_pat (OverLit val False _ ty) mb_neg _  	-- False: Take short cuts only if the literal is not using rebindable syntax  	--   	-- Once that is settled, look for cases where the type of the  @@ -169,7 +175,7 @@ tidyNPat (OverLit val False _ ty) mb_neg _    | isWordTy ty,   Just int_lit <- mb_int_lit = mk_con_pat wordDataCon   (HsWordPrim   int_lit)    | isFloatTy ty,  Just rat_lit <- mb_rat_lit = mk_con_pat floatDataCon  (HsFloatPrim  rat_lit)    | isDoubleTy ty, Just rat_lit <- mb_rat_lit = mk_con_pat doubleDataCon (HsDoublePrim rat_lit) -  | isStringTy ty, Just str_lit <- mb_str_lit = tidyLitPat (HsString str_lit) +  | isStringTy ty, Just str_lit <- mb_str_lit = tidy_lit_pat (HsString str_lit)    where      mk_con_pat :: DataCon -> HsLit -> Pat Id      mk_con_pat con lit = unLoc (mkPrefixConPat con [noLoc $ LitPat lit] ty) @@ -193,7 +199,7 @@ tidyNPat (OverLit val False _ ty) mb_neg _  		   (Nothing, HsIsString s) -> Just s  		   _ -> Nothing -tidyNPat over_lit mb_neg eq  +tidyNPat _ over_lit mb_neg eq     = NPat over_lit mb_neg eq  \end{code} diff --git a/compiler/ghc.cabal.in b/compiler/ghc.cabal.in index b3b4069f80..f70a1b32b0 100644 --- a/compiler/ghc.cabal.in +++ b/compiler/ghc.cabal.in @@ -36,11 +36,6 @@ Flag ghci      Default: False      Manual: True -Flag ncg -    Description: Build the NCG. -    Default: False -    Manual: True -  Flag stage1      Description: Is this stage 1?      Default: False @@ -88,9 +83,6 @@ Library          CPP-Options: -DGHCI          Include-Dirs: ../libffi/build/include -    if !flag(ncg) -        CPP-Options: -DOMIT_NATIVE_CODEGEN -      Build-Depends: bin-package-db      Build-Depends: hoopl @@ -492,10 +484,7 @@ Library          Vectorise.Exp          Vectorise -    -- We only need to expose more modules as some of the ncg code is used -    -- by the LLVM backend so its always included -    if flag(ncg) -        Exposed-Modules: +    Exposed-Modules:              AsmCodeGen              TargetReg              NCGMonad @@ -505,10 +494,6 @@ Library              RegClass              PIC              Platform -            Alpha.Regs -            Alpha.RegInfo -            Alpha.Instr -            Alpha.CodeGen              X86.Regs              X86.RegInfo              X86.Instr diff --git a/compiler/ghc.mk b/compiler/ghc.mk index e26149c902..55ebb84ac9 100644 --- a/compiler/ghc.mk +++ b/compiler/ghc.mk @@ -96,6 +96,58 @@ compiler/stage%/build/Config.hs : mk/config.mk mk/project.mk | $$(dir $$@)/.  	@echo '#error Unknown target arch'                                  >> $@  	@echo '#endif'                                                      >> $@  	@echo                                                               >> $@ +# Sync this with checkOS in configure.ac +	@echo 'cTargetOS :: OS'                                             >> $@ +	@echo '#if linux_TARGET_OS'                                         >> $@ +	@echo 'cTargetOS = Linux'                                           >> $@ +	@echo '#elif freebsd_TARGET_OS'                                     >> $@ +	@echo 'cTargetOS = FreeBSD'                                         >> $@ +	@echo '#elif netbsd_TARGET_OS'                                      >> $@ +	@echo 'cTargetOS = NetBSD'                                          >> $@ +	@echo '#elif openbsd_TARGET_OS'                                     >> $@ +	@echo 'cTargetOS = OpenBSD'                                         >> $@ +	@echo '#elif dragonfly_TARGET_OS'                                   >> $@ +	@echo 'cTargetOS = OtherOS "dragonfly"'                             >> $@ +	@echo '#elif osf1_TARGET_OS'                                        >> $@ +	@echo 'cTargetOS = OtherOS "osf"'                                   >> $@ +	@echo '#elif osf3_TARGET_OS'                                        >> $@ +	@echo 'cTargetOS = OtherOS "osf"'                                   >> $@ +	@echo '#elif hpux_TARGET_OS'                                        >> $@ +	@echo 'cTargetOS = HPUX'                                            >> $@ +	@echo '#elif linuxaout_TARGET_OS'                                   >> $@ +	@echo 'cTargetOS = Linux'                                           >> $@ +	@echo '#elif kfreebsdgnu_TARGET_OS'                                 >> $@ +	@echo 'cTargetOS = OtherOS "kfreebsdgnu"'                           >> $@ +	@echo '#elif freebsd2_TARGET_OS'                                    >> $@ +	@echo 'cTargetOS = FreeBSD'                                         >> $@ +	@echo '#elif solaris2_TARGET_OS'                                    >> $@ +	@echo 'cTargetOS = Solaris'                                         >> $@ +	@echo '#elif cygwin32_TARGET_OS'                                    >> $@ +	@echo 'cTargetOS = Windows'                                         >> $@ +	@echo '#elif mingw32_TARGET_OS'                                     >> $@ +	@echo 'cTargetOS = Windows'                                         >> $@ +	@echo '#elif darwin_TARGET_OS'                                      >> $@ +	@echo 'cTargetOS = OSX'                                             >> $@ +	@echo '#elif gnu_TARGET_OS'                                         >> $@ +	@echo 'cTargetOS = OtherOS "gnu"'                                   >> $@ +	@echo '#elif nextstep2_TARGET_OS'                                   >> $@ +	@echo 'cTargetOS = OtherOS "nextstep"'                              >> $@ +	@echo '#elif nextstep3_TARGET_OS'                                   >> $@ +	@echo 'cTargetOS = OtherOS "nextstep"'                              >> $@ +	@echo '#elif sunos4_TARGET_OS'                                      >> $@ +	@echo 'cTargetOS = Solaris'                                         >> $@ +	@echo '#elif ultrix_TARGET_OS'                                      >> $@ +	@echo 'cTargetOS = OtherOS "ultrix"'                                >> $@ +	@echo '#elif irix_TARGET_OS'                                        >> $@ +	@echo 'cTargetOS = IRIX'                                            >> $@ +	@echo '#elif aix_TARGET_OS'                                         >> $@ +	@echo 'cTargetOS = AIX'                                             >> $@ +	@echo '#elif haiku_TARGET_OS'                                       >> $@ +	@echo 'cTargetOS = OtherOS "haiku"'                                 >> $@ +	@echo '#else'                                                       >> $@ +	@echo '#error Unknown target OS'                                    >> $@ +	@echo '#endif'                                                      >> $@ +	@echo                                                               >> $@  	@echo 'cProjectName          :: String'                             >> $@  	@echo 'cProjectName          = "$(ProjectName)"'                    >> $@  	@echo 'cProjectVersion       :: String'                             >> $@ @@ -371,12 +423,6 @@ endif  endif -ifeq "$(GhcWithNativeCodeGen)" "NO" -# XXX This should logically be a CPP option, but there doesn't seem to -# be a flag for that -compiler_CONFIGURE_OPTS += --ghc-option=-DOMIT_NATIVE_CODEGEN -endif -  ifeq "$(TargetOS_CPP)" "openbsd"  compiler_CONFIGURE_OPTS += --ld-options=-E  endif diff --git a/compiler/ghci/ByteCodeAsm.lhs b/compiler/ghci/ByteCodeAsm.lhs index dfc77e51b2..2c7473b80c 100644 --- a/compiler/ghci/ByteCodeAsm.lhs +++ b/compiler/ghci/ByteCodeAsm.lhs @@ -31,6 +31,7 @@ import Constants  import FastString  import SMRep  import Outputable +import Config  import Control.Monad    ( foldM )  import Control.Monad.ST ( runST ) @@ -44,6 +45,7 @@ import Data.Char        ( ord )  import Data.List  import Data.Map (Map)  import qualified Data.Map as Map +import Distribution.System  import GHC.Base         ( ByteArray#, MutableByteArray#, RealWorld ) @@ -395,12 +397,11 @@ mkBits findLabel st proto_insns            = do st_l1 <- addToSS st_l0 (BCONPtrItbl (getName dcon))                 return (sizeSS16 st_l0, (st_i0,st_l1,st_p0)) -#ifdef mingw32_TARGET_OS         literal st (MachLabel fs (Just sz) _) +        | cTargetOS == Windows              = litlabel st (appendFS fs (mkFastString ('@':show sz)))          -- On Windows, stdcall labels have a suffix indicating the no. of          -- arg words, e.g. foo@8.  testcase: ffi012(ghci) -#endif         literal st (MachLabel fs _ _) = litlabel st fs         literal st (MachWord w)     = int st (fromIntegral w)         literal st (MachInt j)      = int st (fromIntegral j) diff --git a/compiler/hsSyn/Convert.lhs b/compiler/hsSyn/Convert.lhs index b5e6c4129e..5933e9d5fa 100644 --- a/compiler/hsSyn/Convert.lhs +++ b/compiler/hsSyn/Convert.lhs @@ -522,12 +522,15 @@ cvtHsDo do_or_lc stmts    | null stmts = failWith (ptext (sLit "Empty stmt list in do-block"))    | otherwise    = do	{ stmts' <- cvtStmts stmts -	; body <- case last stmts' of -		    L _ (ExprStmt body _ _) -> return body -                    stmt' -> failWith (bad_last stmt') -	; return $ HsDo do_or_lc (init stmts') body void } +        ; let Just (stmts'', last') = snocView stmts' +         +	; last'' <- case last' of +		      L loc (ExprStmt body _ _ _) -> return (L loc (mkLastStmt body)) +                      _ -> failWith (bad_last last') + +	; return $ HsDo do_or_lc (stmts'' ++ [last'']) void }    where -    bad_last stmt = vcat [ ptext (sLit "Illegal last statement of") <+> pprStmtContext do_or_lc <> colon +    bad_last stmt = vcat [ ptext (sLit "Illegal last statement of") <+> pprAStmtContext do_or_lc <> colon                           , nest 2 $ Outputable.ppr stmt  			 , ptext (sLit "(It should be an expression.)") ] @@ -539,7 +542,7 @@ cvtStmt (NoBindS e)    = do { e' <- cvtl e; returnL $ mkExprStmt e' }  cvtStmt (TH.BindS p e) = do { p' <- cvtPat p; e' <- cvtl e; returnL $ mkBindStmt p' e' }  cvtStmt (TH.LetS ds)   = do { ds' <- cvtLocalDecs (ptext (sLit "a let binding")) ds                              ; returnL $ LetStmt ds' } -cvtStmt (TH.ParS dss)  = do { dss' <- mapM cvt_one dss; returnL $ ParStmt dss' } +cvtStmt (TH.ParS dss)  = do { dss' <- mapM cvt_one dss; returnL $ ParStmt dss' noSyntaxExpr noSyntaxExpr noSyntaxExpr }  		       where  			 cvt_one ds = do { ds' <- cvtStmts ds; return (ds', undefined) } diff --git a/compiler/hsSyn/HsExpr.lhs b/compiler/hsSyn/HsExpr.lhs index 06616f16d9..9c88783dd2 100644 --- a/compiler/hsSyn/HsExpr.lhs +++ b/compiler/hsSyn/HsExpr.lhs @@ -23,6 +23,8 @@ import Name  import BasicTypes  import DataCon  import SrcLoc +import Util( dropTail ) +import StaticFlags( opt_PprStyle_Debug )  import Outputable  import FastString @@ -146,8 +148,6 @@ data HsExpr id                                       -- because in this context we never use                                       -- the PatGuard or ParStmt variant                  [LStmt id]           -- "do":one or more stmts -                (LHsExpr id)         -- The body; the last expression in the -                                     -- 'do' of [ body | ... ] in a list comp                  PostTcType           -- Type of the whole expression    | ExplicitList                -- syntactic list @@ -439,7 +439,7 @@ ppr_expr (HsLet binds expr)    = sep [hang (ptext (sLit "let")) 2 (pprBinds binds),           hang (ptext (sLit "in"))  2 (ppr expr)] -ppr_expr (HsDo do_or_list_comp stmts body _) = pprDo do_or_list_comp stmts body +ppr_expr (HsDo do_or_list_comp stmts _) = pprDo do_or_list_comp stmts  ppr_expr (ExplicitList _ exprs)    = brackets (pprDeeperList fsep (punctuate comma (map ppr_lexpr exprs))) @@ -575,7 +575,7 @@ pprParendExpr expr        HsPar {}          -> pp_as_was        HsBracket {}      -> pp_as_was        HsBracketOut _ [] -> pp_as_was -      HsDo sc _ _ _ +      HsDo sc _ _         | isListCompExpr sc -> pp_as_was        _                    -> parens pp_as_was @@ -830,51 +830,59 @@ type LStmtLR idL idR = Located (StmtLR idL idR)  type Stmt id = StmtLR id id --- The SyntaxExprs in here are used *only* for do-notation, which --- has rebindable syntax.  Otherwise they are unused. +-- The SyntaxExprs in here are used *only* for do-notation and monad +-- comprehensions, which have rebindable syntax. Otherwise they are unused.  data StmtLR idL idR -  = BindStmt (LPat idL) +  = LastStmt  -- Always the last Stmt in ListComp, MonadComp, PArrComp,  +    	      -- and (after the renamer) DoExpr, MDoExpr +              -- Not used for GhciStmt, PatGuard, which scope over other stuff +               (LHsExpr idR) +               (SyntaxExpr idR)   -- The return operator, used only for MonadComp +	       		   	  -- For ListComp, PArrComp, we use the baked-in 'return' +				  -- For DoExpr, MDoExpr, we don't appply a 'return' at all +	       		   	  -- See Note [Monad Comprehensions] +  | BindStmt (LPat idL)               (LHsExpr idR) -             (SyntaxExpr idR) -- The (>>=) operator +             (SyntaxExpr idR) -- The (>>=) operator; see Note [The type of bind]               (SyntaxExpr idR) -- The fail operator               -- The fail operator is noSyntaxExpr               -- if the pattern match can't fail    | ExprStmt (LHsExpr idR)     -- See Note [ExprStmt]               (SyntaxExpr idR) -- The (>>) operator +             (SyntaxExpr idR) -- The `guard` operator; used only in MonadComp +                              -- See notes [Monad Comprehensions]               PostTcType       -- Element type of the RHS (used for arrows)    | LetStmt  (HsLocalBindsLR idL idR) -  -- ParStmts only occur in a list comprehension +  -- ParStmts only occur in a list/monad comprehension    | ParStmt  [([LStmt idL], [idR])] -  -- After renaming, the ids are the binders bound by the stmts and used -  -- after them - -  -- "qs, then f by e" ==> TransformStmt qs binders f (Just e) -  -- "qs, then f"      ==> TransformStmt qs binders f Nothing -  | TransformStmt  -         [LStmt idL]	-- Stmts are the ones to the left of the 'then' - -         [idR] 		-- After renaming, the IDs are the binders occurring  -		        -- within this transform statement that are used after it - -         (LHsExpr idR)		-- "then f" - -         (Maybe (LHsExpr idR))	-- "by e" (optional) - -  | GroupStmt  -         [LStmt idL]      -- Stmts to the *left* of the 'group' -	 	       	  -- which generates the tuples to be grouped - -         [(idR, idR)]	  -- See Note [GroupStmt binder map] +             (SyntaxExpr idR)           -- Polymorphic `mzip` for monad comprehensions +             (SyntaxExpr idR)           -- The `>>=` operator +             (SyntaxExpr idR)           -- Polymorphic `return` operator +	     		 		-- with type (forall a. a -> m a) +                                        -- See notes [Monad Comprehensions] +  	    -- After renaming, the ids are the binders  +  	    -- bound by the stmts and used after themp + +  | TransStmt { +      trS_form  :: TransForm, +      trS_stmts :: [LStmt idL],      -- Stmts to the *left* of the 'group' +	            	              -- which generates the tuples to be grouped + +      trS_bndrs :: [(idR, idR)],     -- See Note [TransStmt binder map] -         (Maybe (LHsExpr idR)) 	-- "by e" (optional) +      trS_using :: LHsExpr idR, +      trS_by :: Maybe (LHsExpr idR), 	-- "by e" (optional) +	-- Invariant: if trS_form = GroupBy, then grp_by = Just e -         (Either		-- "using f" -             (LHsExpr idR)	--   Left f  => explicit "using f" -             (SyntaxExpr idR))	--   Right f => implicit; filled in with 'groupWith' -							 +      trS_ret :: SyntaxExpr idR,      -- The monomorphic 'return' function for  +                                       -- the inner monad comprehensions +      trS_bind :: SyntaxExpr idR,     -- The '(>>=)' operator +      trS_fmap :: SyntaxExpr idR      -- The polymorphic 'fmap' function for desugaring +      		   	      	       -- Only for 'group' forms +    }                                  -- See Note [Monad Comprehensions]    -- Recursive statement (see Note [How RecStmt works] below)    | RecStmt @@ -905,20 +913,44 @@ data StmtLR idL idR                                       -- because the Id may be *polymorphic*, but                                       -- the returned thing has to be *monomorphic*,   				     -- so they may be type applications + +      , recS_ret_ty :: PostTcType    -- The type of of do { stmts; return (a,b,c) } +      		       		     -- With rebindable syntax the type might not +				     -- be quite as simple as (m (tya, tyb, tyc)).        }    deriving (Data, Typeable) + +data TransForm		-- The 'f' below is the 'using' function, 'e' is the by function +  = ThenForm		-- then f          or    then f by e +  | GroupFormU		-- group using f   or    group using f by e +  | GroupFormB    	-- group by e   +      -- In the GroupByFormB, trS_using is filled in with +      --    'groupWith' (list comprehensions) or  +      --    'groupM' (monad comprehensions) +  deriving (Data, Typeable)  \end{code} -Note [GroupStmt binder map] +Note [The type of bind in Stmts] +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +Some Stmts, notably BindStmt, keep the (>>=) bind operator.   +We do NOT assume that it has type   +    (>>=) :: m a -> (a -> m b) -> m b +In some cases (see Trac #303, #1537) it might have a more  +exotic type, such as +    (>>=) :: m i j a -> (a -> m j k b) -> m i k b +So we must be careful not to make assumptions about the type. +In particular, the monad may not be uniform throughout. + +Note [TransStmt binder map]  ~~~~~~~~~~~~~~~~~~~~~~~~~~~ -The [(idR,idR)] in a GroupStmt behaves as follows: +The [(idR,idR)] in a TransStmt behaves as follows:    * Before renaming: []    * After renaming:       	  [ (x27,x27), ..., (z35,z35) ]      These are the variables  -        bound by the stmts to the left of the 'group' +       bound by the stmts to the left of the 'group'         and used either in the 'by' clause,                   or     in the stmts following the 'group'      Each item is a pair of identical variables. @@ -952,7 +984,13 @@ depends on the context.  Consider the following contexts:                  E :: Bool            Translation: if E then fail else ... -Array comprehensions are handled like list comprehensions -=chak +        A monad comprehension of type (m res_ty) +        ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +        * ExprStmt E Bool:   [ .. | .... E ] +                E :: Bool +          Translation: guard E >> ... + +Array comprehensions are handled like list comprehensions.  Note [How RecStmt works]  ~~~~~~~~~~~~~~~~~~~~~~~~ @@ -993,23 +1031,60 @@ A (RecStmt stmts) types as if you had written  where v1..vn are the later_ids        r1..rm are the rec_ids +Note [Monad Comprehensions] +~~~~~~~~~~~~~~~~~~~~~~~~~~~ +Monad comprehensions require separate functions like 'return' and +'>>=' for desugaring. These functions are stored in the statements +used in monad comprehensions. For example, the 'return' of the 'LastStmt' +expression is used to lift the body of the monad comprehension: + +  [ body | stmts ] +   => +  stmts >>= \bndrs -> return body + +In transform and grouping statements ('then ..' and 'then group ..') the +'return' function is required for nested monad comprehensions, for example: + +  [ body | stmts, then f, rest ] +   => +  f [ env | stmts ] >>= \bndrs -> [ body | rest ] + +ExprStmts require the 'Control.Monad.guard' function for boolean +expressions: + +  [ body | exp, stmts ] +   => +  guard exp >> [ body | stmts ] + +Grouping/parallel statements require the 'Control.Monad.Group.groupM' and +'Control.Monad.Zip.mzip' functions: + +  [ body | stmts, then group by e, rest] +   => +  groupM [ body | stmts ] >>= \bndrs -> [ body | rest ] + +  [ body | stmts1 | stmts2 | .. ] +   => +  mzip stmts1 (mzip stmts2 (..)) >>= \(bndrs1, (bndrs2, ..)) -> return body + +In any other context than 'MonadComp', the fields for most of these +'SyntaxExpr's stay bottom. +  \begin{code}  instance (OutputableBndr idL, OutputableBndr idR) => Outputable (StmtLR idL idR) where      ppr stmt = pprStmt stmt  pprStmt :: (OutputableBndr idL, OutputableBndr idR) => (StmtLR idL idR) -> SDoc +pprStmt (LastStmt expr _)         = ifPprDebug (ptext (sLit "[last]")) <+> ppr expr  pprStmt (BindStmt pat expr _ _)   = hsep [ppr pat, ptext (sLit "<-"), ppr expr]  pprStmt (LetStmt binds)           = hsep [ptext (sLit "let"), pprBinds binds] -pprStmt (ExprStmt expr _ _)       = ppr expr -pprStmt (ParStmt stmtss)          = hsep (map doStmts stmtss) +pprStmt (ExprStmt expr _ _ _)     = ppr expr +pprStmt (ParStmt stmtss _ _ _)    = hsep (map doStmts stmtss)    where doStmts stmts = ptext (sLit "| ") <> ppr stmts -pprStmt (TransformStmt stmts bndrs using by) -  = sep (ppr_lc_stmts stmts ++ [pprTransformStmt bndrs using by]) - -pprStmt (GroupStmt stmts _ by using)  -  = sep (ppr_lc_stmts stmts ++ [pprGroupStmt by using]) +pprStmt (TransStmt { trS_stmts = stmts, trS_by = by, trS_using = using, trS_form = form }) +  = sep (ppr_lc_stmts stmts ++ [pprTransStmt by using form])  pprStmt (RecStmt { recS_stmts = segment, recS_rec_ids = rec_ids                   , recS_later_ids = later_ids }) @@ -1024,40 +1099,47 @@ pprTransformStmt bndrs using by          , nest 2 (ppr using)          , nest 2 (pprBy by)] -pprGroupStmt :: OutputableBndr id => Maybe (LHsExpr id) -                                  -> Either (LHsExpr id) (SyntaxExpr is) +pprTransStmt :: OutputableBndr id => Maybe (LHsExpr id) +                                  -> LHsExpr id -> TransForm  				  -> SDoc -pprGroupStmt by using  -  = sep [ ptext (sLit "then group"), nest 2 (pprBy by), nest 2 (ppr_using using)] -  where -    ppr_using (Right _) = empty -    ppr_using (Left e)  = ptext (sLit "using") <+> ppr e +pprTransStmt by using ThenForm +  = sep [ ptext (sLit "then"), nest 2 (ppr using), nest 2 (pprBy by)] +pprTransStmt by _ GroupFormB +  = sep [ ptext (sLit "then group"), nest 2 (pprBy by) ] +pprTransStmt by using GroupFormU +  = sep [ ptext (sLit "then group"), nest 2 (pprBy by), nest 2 (ptext (sLit "using") <+> ppr using)]  pprBy :: OutputableBndr id => Maybe (LHsExpr id) -> SDoc  pprBy Nothing  = empty  pprBy (Just e) = ptext (sLit "by") <+> ppr e -pprDo :: OutputableBndr id => HsStmtContext any -> [LStmt id] -> LHsExpr id -> SDoc -pprDo DoExpr      stmts body = ptext (sLit "do")  <+> ppr_do_stmts stmts body -pprDo GhciStmt    stmts body = ptext (sLit "do")  <+> ppr_do_stmts stmts body -pprDo MDoExpr     stmts body = ptext (sLit "mdo") <+> ppr_do_stmts stmts body -pprDo ListComp    stmts body = brackets    $ pprComp stmts body -pprDo PArrComp    stmts body = pa_brackets $ pprComp stmts body -pprDo _           _     _    = panic "pprDo" -- PatGuard, ParStmtCxt - -ppr_do_stmts :: OutputableBndr id => [LStmt id] -> LHsExpr id -> SDoc +pprDo :: OutputableBndr id => HsStmtContext any -> [LStmt id] -> SDoc +pprDo DoExpr      stmts = ptext (sLit "do")  <+> ppr_do_stmts stmts +pprDo GhciStmt    stmts = ptext (sLit "do")  <+> ppr_do_stmts stmts +pprDo ArrowExpr   stmts = ptext (sLit "do")  <+> ppr_do_stmts stmts +pprDo MDoExpr     stmts = ptext (sLit "mdo") <+> ppr_do_stmts stmts +pprDo ListComp    stmts = brackets    $ pprComp stmts +pprDo PArrComp    stmts = pa_brackets $ pprComp stmts +pprDo MonadComp   stmts = brackets    $ pprComp stmts +pprDo _           _     = panic "pprDo" -- PatGuard, ParStmtCxt + +ppr_do_stmts :: OutputableBndr id => [LStmt id] -> SDoc  -- Print a bunch of do stmts, with explicit braces and semicolons,  -- so that we are not vulnerable to layout bugs -ppr_do_stmts stmts body -  = lbrace <+> pprDeeperList vcat ([ppr s <> semi | s <- stmts] ++ [ppr body]) +ppr_do_stmts stmts  +  = lbrace <+> pprDeeperList vcat (punctuate semi (map ppr stmts))             <+> rbrace  ppr_lc_stmts :: OutputableBndr id => [LStmt id] -> [SDoc]  ppr_lc_stmts stmts = [ppr s <> comma | s <- stmts] -pprComp :: OutputableBndr id => [LStmt id] -> LHsExpr id -> SDoc -pprComp quals body	  -- Prints:  body | qual1, ..., qualn  -  = hang (ppr body <+> char '|') 2 (interpp'SP quals) +pprComp :: OutputableBndr id => [LStmt id] -> SDoc +pprComp quals	  -- Prints:  body | qual1, ..., qualn  +  | not (null quals) +  , L _ (LastStmt body _) <- last quals +  = hang (ppr body <+> char '|') 2 (interpp'SP (dropTail 1 quals)) +  | otherwise +  = pprPanic "pprComp" (interpp'SP quals)  \end{code}  %************************************************************************ @@ -1175,26 +1257,33 @@ data HsMatchContext id  -- Context of a Match  data HsStmtContext id    = ListComp -  | DoExpr -  | GhciStmt				 -- A command-line Stmt in GHCi pat <- rhs -  | MDoExpr                              -- Recursive do-expression +  | MonadComp    | PArrComp                             -- Parallel array comprehension + +  | DoExpr				 -- do { ... } +  | MDoExpr                              -- mdo { ... }  ie recursive do-expression  +  | ArrowExpr				 -- do-notation in an arrow-command context + +  | GhciStmt				 -- A command-line Stmt in GHCi pat <- rhs    | PatGuard (HsMatchContext id)         -- Pattern guard for specified thing    | ParStmtCtxt (HsStmtContext id)       -- A branch of a parallel stmt -  | TransformStmtCtxt (HsStmtContext id) -- A branch of a transform stmt +  | TransStmtCtxt (HsStmtContext id)     -- A branch of a transform stmt    deriving (Data, Typeable)  \end{code}  \begin{code} -isDoExpr :: HsStmtContext id -> Bool -isDoExpr DoExpr  = True -isDoExpr MDoExpr = True -isDoExpr _       = False -  isListCompExpr :: HsStmtContext id -> Bool -isListCompExpr ListComp = True -isListCompExpr PArrComp = True -isListCompExpr _        = False +-- Uses syntax [ e | quals ] +isListCompExpr ListComp  = True +isListCompExpr PArrComp  = True +isListCompExpr MonadComp = True +isListCompExpr _         = False + +isMonadCompExpr :: HsStmtContext id -> Bool +isMonadCompExpr MonadComp            = True +isMonadCompExpr (ParStmtCtxt ctxt)   = isMonadCompExpr ctxt +isMonadCompExpr (TransStmtCtxt ctxt) = isMonadCompExpr ctxt +isMonadCompExpr _                    = False  \end{code}  \begin{code} @@ -1231,33 +1320,41 @@ pprMatchContextNoun ProcExpr        = ptext (sLit "arrow abstraction")  pprMatchContextNoun (StmtCtxt ctxt) = ptext (sLit "pattern binding in")                                        $$ pprStmtContext ctxt -pprStmtContext :: Outputable id => HsStmtContext id -> SDoc +----------------- +pprAStmtContext, pprStmtContext :: Outputable id => HsStmtContext id -> SDoc +pprAStmtContext ctxt = article <+> pprStmtContext ctxt +  where +    pp_an = ptext (sLit "an") +    pp_a  = ptext (sLit "a") +    article = case ctxt of +                  MDoExpr  -> pp_an +                  PArrComp -> pp_an +		  GhciStmt -> pp_an +                  _        -> pp_a + + +----------------- +pprStmtContext GhciStmt        = ptext (sLit "interactive GHCi command") +pprStmtContext DoExpr          = ptext (sLit "'do' block") +pprStmtContext MDoExpr         = ptext (sLit "'mdo' block") +pprStmtContext ArrowExpr       = ptext (sLit "'do' block in an arrow command") +pprStmtContext ListComp        = ptext (sLit "list comprehension") +pprStmtContext MonadComp       = ptext (sLit "monad comprehension") +pprStmtContext PArrComp        = ptext (sLit "array comprehension") +pprStmtContext (PatGuard ctxt) = ptext (sLit "pattern guard for") $$ pprMatchContext ctxt + +-- Drop the inner contexts when reporting errors, else we get +--     Unexpected transform statement +--     in a transformed branch of +--          transformed branch of +--          transformed branch of monad comprehension  pprStmtContext (ParStmtCtxt c) - = sep [ptext (sLit "a parallel branch of"), pprStmtContext c] -pprStmtContext (TransformStmtCtxt c) - = sep [ptext (sLit "a transformed branch of"), pprStmtContext c] -pprStmtContext (PatGuard ctxt) - = ptext (sLit "a pattern guard for") $$ pprMatchContext ctxt -pprStmtContext GhciStmt        = ptext (sLit "an interactive GHCi command") -pprStmtContext DoExpr          = ptext (sLit "a 'do' expression") -pprStmtContext MDoExpr         = ptext (sLit "an 'mdo' expression") -pprStmtContext ListComp        = ptext (sLit "a list comprehension") -pprStmtContext PArrComp        = ptext (sLit "an array comprehension") - -{- -pprMatchRhsContext (FunRhs fun) = ptext (sLit "a right-hand side of function") <+> quotes (ppr fun) -pprMatchRhsContext CaseAlt      = ptext (sLit "the body of a case alternative") -pprMatchRhsContext PatBindRhs   = ptext (sLit "the right-hand side of a pattern binding") -pprMatchRhsContext LambdaExpr   = ptext (sLit "the body of a lambda") -pprMatchRhsContext ProcExpr     = ptext (sLit "the body of a proc") -pprMatchRhsContext other        = panic "pprMatchRhsContext"    -- RecUpd, StmtCtxt - --- Used for the result statement of comprehension --- e.g. the 'e' in      [ e | ... ] ---      or the 'r' in   f x = r -pprStmtResultContext (PatGuard ctxt) = pprMatchRhsContext ctxt -pprStmtResultContext other           = ptext (sLit "the result of") <+> pprStmtContext other --} + | opt_PprStyle_Debug = sep [ptext (sLit "parallel branch of"), pprAStmtContext c] + | otherwise          = pprStmtContext c +pprStmtContext (TransStmtCtxt c) + | opt_PprStyle_Debug = sep [ptext (sLit "transformed branch of"), pprAStmtContext c] + | otherwise          = pprStmtContext c +  -- Used to generate the string for a *runtime* error message  matchContextErrString :: Outputable id => HsMatchContext id -> SDoc @@ -1268,14 +1365,16 @@ matchContextErrString RecUpd                     = ptext (sLit "record update")  matchContextErrString LambdaExpr                 = ptext (sLit "lambda")  matchContextErrString ProcExpr                   = ptext (sLit "proc")  matchContextErrString ThPatQuote                 = panic "matchContextErrString"  -- Not used at runtime -matchContextErrString (StmtCtxt (ParStmtCtxt c)) = matchContextErrString (StmtCtxt c) -matchContextErrString (StmtCtxt (TransformStmtCtxt c)) = matchContextErrString (StmtCtxt c) -matchContextErrString (StmtCtxt (PatGuard _))    = ptext (sLit "pattern guard") -matchContextErrString (StmtCtxt GhciStmt)        = ptext (sLit "interactive GHCi command") -matchContextErrString (StmtCtxt DoExpr)          = ptext (sLit "'do' expression") -matchContextErrString (StmtCtxt MDoExpr)         = ptext (sLit "'mdo' expression") -matchContextErrString (StmtCtxt ListComp)        = ptext (sLit "list comprehension") -matchContextErrString (StmtCtxt PArrComp)        = ptext (sLit "array comprehension") +matchContextErrString (StmtCtxt (ParStmtCtxt c))   = matchContextErrString (StmtCtxt c) +matchContextErrString (StmtCtxt (TransStmtCtxt c)) = matchContextErrString (StmtCtxt c) +matchContextErrString (StmtCtxt (PatGuard _))      = ptext (sLit "pattern guard") +matchContextErrString (StmtCtxt GhciStmt)          = ptext (sLit "interactive GHCi command") +matchContextErrString (StmtCtxt DoExpr)            = ptext (sLit "'do' block") +matchContextErrString (StmtCtxt ArrowExpr)         = ptext (sLit "'do' block") +matchContextErrString (StmtCtxt MDoExpr)           = ptext (sLit "'mdo' block") +matchContextErrString (StmtCtxt ListComp)          = ptext (sLit "list comprehension") +matchContextErrString (StmtCtxt MonadComp)         = ptext (sLit "monad comprehension") +matchContextErrString (StmtCtxt PArrComp)          = ptext (sLit "array comprehension")  \end{code}  \begin{code} @@ -1286,11 +1385,16 @@ pprMatchInCtxt ctxt match  = hang (ptext (sLit "In") <+> pprMatchContext ctxt <>  pprStmtInCtxt :: (OutputableBndr idL, OutputableBndr idR)     	       => HsStmtContext idL -> StmtLR idL idR -> SDoc -pprStmtInCtxt ctxt stmt = hang (ptext (sLit "In a stmt of") <+> pprStmtContext ctxt <> colon) -		    	  4 (ppr_stmt stmt) +pprStmtInCtxt ctxt (LastStmt e _) +  | isListCompExpr ctxt      -- For [ e | .. ], do not mutter about "stmts" +  = hang (ptext (sLit "In the expression:")) 2 (ppr e) + +pprStmtInCtxt ctxt stmt  +  = hang (ptext (sLit "In a stmt of") <+> pprAStmtContext ctxt <> colon) +       2 (ppr_stmt stmt)    where      -- For Group and Transform Stmts, don't print the nested stmts! -    ppr_stmt (GroupStmt _ _ by using)         = pprGroupStmt by using -    ppr_stmt (TransformStmt _ bndrs using by) = pprTransformStmt bndrs using by -    ppr_stmt stmt                             = pprStmt stmt +    ppr_stmt (TransStmt { trS_by = by, trS_using = using +                        , trS_form = form }) = pprTransStmt by using form +    ppr_stmt stmt = pprStmt stmt  \end{code} diff --git a/compiler/hsSyn/HsLit.lhs b/compiler/hsSyn/HsLit.lhs index 0874dda858..4a565ff8ba 100644 --- a/compiler/hsSyn/HsLit.lhs +++ b/compiler/hsSyn/HsLit.lhs @@ -63,8 +63,7 @@ instance Eq HsLit where  data HsOverLit id 	-- An overloaded literal    = OverLit {  	ol_val :: OverLitVal,  -	ol_rebindable :: Bool,		-- True <=> rebindable syntax -					-- False <=> standard syntax +	ol_rebindable :: Bool,		-- Note [ol_rebindable]  	ol_witness :: SyntaxExpr id,	-- Note [Overloaded literal witnesses]  	ol_type :: PostTcType }    deriving (Data, Typeable) @@ -79,6 +78,19 @@ overLitType :: HsOverLit a -> Type  overLitType = ol_type  \end{code} +Note [ol_rebindable] +~~~~~~~~~~~~~~~~~~~~ +The ol_rebindable field is True if this literal is actually  +using rebindable syntax.  Specifically: + +  False iff ol_witness is the standard one +  True  iff ol_witness is non-standard + +Equivalently it's True if +  a) RebindableSyntax is on +  b) the witness for fromInteger/fromRational/fromString +     that happens to be in scope isn't the standard one +  Note [Overloaded literal witnesses]  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~  *Before* type checking, the SyntaxExpr in an HsOverLit is the @@ -89,7 +101,7 @@ This witness should replace the literal.  This dual role is unusual, because we're replacing 'fromInteger' with   a call to fromInteger.  Reason: it allows commoning up of the fromInteger -calls, which wouldn't be possible if the desguarar made the application +calls, which wouldn't be possible if the desguarar made the application.  The PostTcType in each branch records the type the overload literal is  found to have. diff --git a/compiler/hsSyn/HsPat.lhs b/compiler/hsSyn/HsPat.lhs index 740bfa7172..1098ff03b2 100644 --- a/compiler/hsSyn/HsPat.lhs +++ b/compiler/hsSyn/HsPat.lhs @@ -122,7 +122,9 @@ data Pat id    | LitPat	    HsLit		-- Used for *non-overloaded* literal patterns:  					-- Int#, Char#, Int, Char, String, etc. -  | NPat	    (HsOverLit id)		-- ALWAYS positive +  | NPat		-- Used for all overloaded literals,  +    			-- including overloaded strings with -XOverloadedStrings +                    (HsOverLit id)		-- ALWAYS positive  		    (Maybe (SyntaxExpr id))	-- Just (Name of 'negate') for negative  						-- patterns, Nothing otherwise  		    (SyntaxExpr id)		-- Equality checker, of type t->t->Bool diff --git a/compiler/hsSyn/HsUtils.lhs b/compiler/hsSyn/HsUtils.lhs index 3316634e87..d86b6323cd 100644 --- a/compiler/hsSyn/HsUtils.lhs +++ b/compiler/hsSyn/HsUtils.lhs @@ -21,7 +21,7 @@ module HsUtils(    mkMatchGroup, mkMatch, mkHsLam, mkHsIf,    mkHsWrap, mkLHsWrap, mkHsWrapCo, mkLHsWrapCo,    coToHsWrapper, mkHsDictLet, mkHsLams, -  mkHsOpApp, mkHsDo, mkHsWrapPat, mkHsWrapPatCo, +  mkHsOpApp, mkHsDo, mkHsComp, mkHsWrapPat, mkHsWrapPatCo,    nlHsTyApp, nlHsVar, nlHsLit, nlHsApp, nlHsApps, nlHsIntLit, nlHsVarApps,     nlHsDo, nlHsOpApp, nlHsLam, nlHsPar, nlHsIf, nlHsCase, nlList, @@ -42,8 +42,8 @@ module HsUtils(    nlHsAppTy, nlHsTyVar, nlHsFunTy, nlHsTyConApp,     -- Stmts -  mkTransformStmt, mkTransformByStmt, mkExprStmt, mkBindStmt, -  mkGroupUsingStmt, mkGroupByStmt, mkGroupByUsingStmt,  +  mkTransformStmt, mkTransformByStmt, mkExprStmt, mkBindStmt, mkLastStmt, +  emptyTransStmt, mkGroupUsingStmt, mkGroupByStmt, mkGroupByUsingStmt,     emptyRecStmt, mkRecStmt,     -- Template Haskell @@ -190,14 +190,13 @@ mkSimpleHsAlt pat expr  mkHsIntegral   :: Integer -> PostTcType -> HsOverLit id  mkHsFractional :: Rational -> PostTcType -> HsOverLit id  mkHsIsString   :: FastString -> PostTcType -> HsOverLit id -mkHsDo         :: HsStmtContext Name -> [LStmt id] -> LHsExpr id -> HsExpr id +mkHsDo         :: HsStmtContext Name -> [LStmt id] -> HsExpr id +mkHsComp       :: HsStmtContext Name -> [LStmt id] -> LHsExpr id -> HsExpr id  mkNPat      :: HsOverLit id -> Maybe (SyntaxExpr id) -> Pat id  mkNPlusKPat :: Located id -> HsOverLit id -> Pat id -mkTransformStmt   :: [LStmt idL] -> LHsExpr idR                -> StmtLR idL idR -mkTransformByStmt :: [LStmt idL] -> LHsExpr idR -> LHsExpr idR -> StmtLR idL idR - +mkLastStmt :: LHsExpr idR -> StmtLR idL idR  mkExprStmt :: LHsExpr idR -> StmtLR idL idR  mkBindStmt :: LPat idL -> LHsExpr idR -> StmtLR idL idR @@ -212,7 +211,10 @@ mkHsIsString   s       = OverLit (HsIsString   s)  noRebindableInfo noSyntaxExpr  noRebindableInfo :: Bool  noRebindableInfo = error "noRebindableInfo" 	-- Just another placeholder;  -mkHsDo ctxt stmts body = HsDo ctxt stmts body placeHolderType +mkHsDo ctxt stmts = HsDo ctxt stmts placeHolderType +mkHsComp ctxt stmts expr = mkHsDo ctxt (stmts ++ [last_stmt]) +  where +    last_stmt = L (getLoc expr) $ mkLastStmt expr  mkHsIf :: LHsExpr id -> LHsExpr id -> LHsExpr id -> HsExpr id  mkHsIf c a b = HsIf (Just noSyntaxExpr) c a b @@ -220,24 +222,32 @@ mkHsIf c a b = HsIf (Just noSyntaxExpr) c a b  mkNPat lit neg     = NPat lit neg noSyntaxExpr  mkNPlusKPat id lit = NPlusKPat id lit noSyntaxExpr noSyntaxExpr -mkTransformStmt   stmts usingExpr        = TransformStmt stmts [] usingExpr Nothing -mkTransformByStmt stmts usingExpr byExpr = TransformStmt stmts [] usingExpr (Just byExpr) - +mkTransformStmt   :: [LStmt idL] -> LHsExpr idR                -> StmtLR idL idR +mkTransformByStmt :: [LStmt idL] -> LHsExpr idR -> LHsExpr idR -> StmtLR idL idR  mkGroupUsingStmt   :: [LStmt idL]                -> LHsExpr idR -> StmtLR idL idR  mkGroupByStmt      :: [LStmt idL] -> LHsExpr idR                -> StmtLR idL idR  mkGroupByUsingStmt :: [LStmt idL] -> LHsExpr idR -> LHsExpr idR -> StmtLR idL idR -mkGroupUsingStmt   stmts usingExpr        = GroupStmt stmts [] Nothing       (Left usingExpr)     -mkGroupByStmt      stmts byExpr           = GroupStmt stmts [] (Just byExpr) (Right noSyntaxExpr) -mkGroupByUsingStmt stmts byExpr usingExpr = GroupStmt stmts [] (Just byExpr) (Left usingExpr)     - -mkExprStmt expr	    = ExprStmt expr noSyntaxExpr placeHolderType +emptyTransStmt :: StmtLR idL idR +emptyTransStmt = TransStmt { trS_form = undefined, trS_stmts = [], trS_bndrs = []  +                           , trS_by = Nothing, trS_using = noLoc noSyntaxExpr +                           , trS_ret = noSyntaxExpr, trS_bind = noSyntaxExpr +                           , trS_fmap = noSyntaxExpr } +mkTransformStmt   ss u    = emptyTransStmt { trS_form = ThenForm, trS_stmts = ss, trS_using = u } +mkTransformByStmt ss u b  = emptyTransStmt { trS_form = ThenForm, trS_stmts = ss, trS_using = u, trS_by = Just b } +mkGroupByStmt      ss b   = emptyTransStmt { trS_form = GroupFormB, trS_stmts = ss, trS_by = Just b } +mkGroupUsingStmt   ss u   = emptyTransStmt { trS_form = GroupFormU, trS_stmts = ss, trS_using = u } +mkGroupByUsingStmt ss b u = emptyTransStmt { trS_form = GroupFormU, trS_stmts = ss +                                           , trS_by = Just b, trS_using = u } + +mkLastStmt expr	    = LastStmt expr noSyntaxExpr +mkExprStmt expr	    = ExprStmt expr noSyntaxExpr noSyntaxExpr placeHolderType  mkBindStmt pat expr = BindStmt pat expr noSyntaxExpr noSyntaxExpr  emptyRecStmt = RecStmt { recS_stmts = [], recS_later_ids = [], recS_rec_ids = []                         , recS_ret_fn = noSyntaxExpr, recS_mfix_fn = noSyntaxExpr  		       , recS_bind_fn = noSyntaxExpr -                       , recS_rec_rets = [] } +                       , recS_rec_rets = [], recS_ret_ty = placeHolderType }  mkRecStmt stmts = emptyRecStmt { recS_stmts = stmts } @@ -327,8 +337,8 @@ nlWildConPat con = noLoc (ConPatIn (noLoc (getRdrName con))  nlWildPat :: LPat id  nlWildPat  = noLoc (WildPat placeHolderType)	-- Pre-typechecking -nlHsDo :: HsStmtContext Name -> [LStmt id] -> LHsExpr id -> LHsExpr id -nlHsDo ctxt stmts body = noLoc (mkHsDo ctxt stmts body) +nlHsDo :: HsStmtContext Name -> [LStmt id] -> LHsExpr id +nlHsDo ctxt stmts = noLoc (mkHsDo ctxt stmts)  nlHsOpApp :: LHsExpr id -> id -> LHsExpr id -> LHsExpr id  nlHsOpApp e1 op e2 = noLoc (mkHsOpApp e1 op e2) @@ -496,12 +506,12 @@ collectStmtBinders :: StmtLR idL idR -> [idL]    -- Id Binders for a Stmt... [but what about pattern-sig type vars]?  collectStmtBinders (BindStmt pat _ _ _) = collectPatBinders pat  collectStmtBinders (LetStmt binds)      = collectLocalBinders binds -collectStmtBinders (ExprStmt _ _ _)     = [] -collectStmtBinders (ParStmt xs)         = collectLStmtsBinders +collectStmtBinders (ExprStmt {})        = [] +collectStmtBinders (LastStmt {})        = [] +collectStmtBinders (ParStmt xs _ _ _)   = collectLStmtsBinders                                          $ concatMap fst xs -collectStmtBinders (TransformStmt stmts _ _ _)   = collectLStmtsBinders stmts -collectStmtBinders (GroupStmt     stmts _ _ _)   = collectLStmtsBinders stmts -collectStmtBinders (RecStmt { recS_stmts = ss }) = collectLStmtsBinders ss +collectStmtBinders (TransStmt { trS_stmts = stmts }) = collectLStmtsBinders stmts +collectStmtBinders (RecStmt { recS_stmts = ss })     = collectLStmtsBinders ss  ----------------- Patterns -------------------------- @@ -642,12 +652,12 @@ lStmtsImplicits = hs_lstmts      hs_stmt (BindStmt pat _ _ _) = lPatImplicits pat      hs_stmt (LetStmt binds)      = hs_local_binds binds -    hs_stmt (ExprStmt _ _ _)     = emptyNameSet -    hs_stmt (ParStmt xs)         = hs_lstmts $ concatMap fst xs +    hs_stmt (ExprStmt {})        = emptyNameSet +    hs_stmt (LastStmt {})        = emptyNameSet +    hs_stmt (ParStmt xs _ _ _)   = hs_lstmts $ concatMap fst xs -    hs_stmt (TransformStmt stmts _ _ _)   = hs_lstmts stmts -    hs_stmt (GroupStmt     stmts _ _ _)   = hs_lstmts stmts -    hs_stmt (RecStmt { recS_stmts = ss }) = hs_lstmts ss +    hs_stmt (TransStmt { trS_stmts = stmts }) = hs_lstmts stmts +    hs_stmt (RecStmt { recS_stmts = ss })     = hs_lstmts ss      hs_local_binds (HsValBinds val_binds) = hsValBindsImplicits val_binds      hs_local_binds (HsIPBinds _)         = emptyNameSet diff --git a/compiler/iface/IfaceSyn.lhs b/compiler/iface/IfaceSyn.lhs index 48bef49f1e..ef0ef5c5f0 100644 --- a/compiler/iface/IfaceSyn.lhs +++ b/compiler/iface/IfaceSyn.lhs @@ -5,34 +5,34 @@  \begin{code}  module IfaceSyn ( -	module IfaceType,		-- Re-export all this +        module IfaceType, -	IfaceDecl(..), IfaceClassOp(..), IfaceConDecl(..), IfaceConDecls(..), -	IfaceExpr(..), IfaceAlt, IfaceNote(..), IfaceLetBndr(..), -	IfaceBinding(..), IfaceConAlt(..),  -	IfaceIdInfo(..), IfaceIdDetails(..), IfaceUnfolding(..), -	IfaceInfoItem(..), IfaceRule(..), IfaceAnnotation(..), IfaceAnnTarget, -	IfaceInst(..), IfaceFamInst(..), +        IfaceDecl(..), IfaceClassOp(..), IfaceConDecl(..), IfaceConDecls(..), +        IfaceExpr(..), IfaceAlt, IfaceNote(..), IfaceLetBndr(..), +        IfaceBinding(..), IfaceConAlt(..), +        IfaceIdInfo(..), IfaceIdDetails(..), IfaceUnfolding(..), +        IfaceInfoItem(..), IfaceRule(..), IfaceAnnotation(..), IfaceAnnTarget, +        IfaceInst(..), IfaceFamInst(..), -	-- Misc +        -- Misc          ifaceDeclSubBndrs, visibleIfConDecls,          -- Free Names          freeNamesIfDecl, freeNamesIfRule, -	-- Pretty printing -	pprIfaceExpr, pprIfaceDeclHead  +        -- Pretty printing +        pprIfaceExpr, pprIfaceDeclHead      ) where  #include "HsVersions.h"  import IfaceType  import CoreSyn( DFunArg, dfunArgExprs ) -import PprCore()            -- Printing DFunArgs +import PprCore()     -- Printing DFunArgs  import Demand  import Annotations  import Class -import NameSet  +import NameSet  import Name  import CostCentre  import Literal @@ -48,74 +48,75 @@ infixl 3 &&&  %************************************************************************ -%*									* -		Data type declarations -%*									* +%*                                                                      * +    Data type declarations +%*                                                                      *  %************************************************************************  \begin{code} -data IfaceDecl  -  = IfaceId { ifName   	  :: OccName, -	      ifType   	  :: IfaceType,  -	      ifIdDetails :: IfaceIdDetails, -	      ifIdInfo    :: IfaceIdInfo } - -  | IfaceData { ifName       :: OccName,	-- Type constructor -		ifTyVars     :: [IfaceTvBndr],	-- Type variables -		ifCtxt	     :: IfaceContext,	-- The "stupid theta" -		ifCons	     :: IfaceConDecls,	-- Includes new/data info -	        ifRec	     :: RecFlag,	-- Recursive or not? -		ifGadtSyntax :: Bool,		-- True <=> declared using -						-- GADT syntax  -		ifGeneric    :: Bool,		-- True <=> generic converter -						--          functions available -    						-- We need this for imported -    						-- data decls, since the -    						-- imported modules may have -    						-- been compiled with -    						-- different flags to the -    						-- current compilation unit  +data IfaceDecl +  = IfaceId { ifName      :: OccName, +              ifType      :: IfaceType, +              ifIdDetails :: IfaceIdDetails, +              ifIdInfo    :: IfaceIdInfo } + +  | IfaceData { ifName       :: OccName,        -- Type constructor +                ifTyVars     :: [IfaceTvBndr],  -- Type variables +                ifCtxt       :: IfaceContext,   -- The "stupid theta" +                ifCons       :: IfaceConDecls,  -- Includes new/data info +                ifRec        :: RecFlag,        -- Recursive or not? +                ifGadtSyntax :: Bool,           -- True <=> declared using +                                                -- GADT syntax +                ifGeneric    :: Bool,           -- True <=> generic converter +                                                --          functions available +                                                -- We need this for imported +                                                -- data decls, since the +                                                -- imported modules may have +                                                -- been compiled with +                                                -- different flags to the +                                                -- current compilation unit                  ifFamInst    :: Maybe (IfaceTyCon, [IfaceType])                                                  -- Just <=> instance of family -                                                -- Invariant:  +                                                -- Invariant:                                                  --   ifCons /= IfOpenDataTyCon                                                  --   for family instances      } -  | IfaceSyn  {	ifName    :: OccName,		-- Type constructor -		ifTyVars  :: [IfaceTvBndr],	-- Type variables -		ifSynKind :: IfaceKind,		-- Kind of the *rhs* (not of the tycon) -		ifSynRhs  :: Maybe IfaceType,	-- Just rhs for an ordinary synonyn -						-- Nothing for an open family +  | IfaceSyn  { ifName    :: OccName,           -- Type constructor +                ifTyVars  :: [IfaceTvBndr],     -- Type variables +                ifSynKind :: IfaceKind,         -- Kind of the *rhs* (not of the tycon) +                ifSynRhs  :: Maybe IfaceType,   -- Just rhs for an ordinary synonyn +                                                -- Nothing for an open family                  ifFamInst :: Maybe (IfaceTyCon, [IfaceType])                                                  -- Just <=> instance of family                                                  -- Invariant: ifOpenSyn == False                                                  --   for family instances      } -  | IfaceClass { ifCtxt    :: IfaceContext, 	-- Context... -		 ifName    :: OccName,		-- Name of the class -		 ifTyVars  :: [IfaceTvBndr],	-- Type variables -		 ifFDs     :: [FunDep FastString], -- Functional dependencies -		 ifATs	   :: [IfaceDecl],	-- Associated type families -		 ifSigs    :: [IfaceClassOp],	-- Method signatures -	         ifRec	   :: RecFlag		-- Is newtype/datatype associated with the class recursive? +  | IfaceClass { ifCtxt    :: IfaceContext,     -- Context... +                 ifName    :: OccName,          -- Name of the class +                 ifTyVars  :: [IfaceTvBndr],    -- Type variables +                 ifFDs     :: [FunDep FastString], -- Functional dependencies +                 ifATs     :: [IfaceDecl],      -- Associated type families +                 ifSigs    :: [IfaceClassOp],   -- Method signatures +                 ifRec     :: RecFlag           -- Is newtype/datatype associated +                                                --   with the class recursive?      }    | IfaceForeign { ifName :: OccName,           -- Needs expanding when we move                                                  -- beyond .NET -		   ifExtName :: Maybe FastString } +                   ifExtName :: Maybe FastString }  data IfaceClassOp = IfaceClassOp OccName DefMethSpec IfaceType -	-- Nothing    => no default method -	-- Just False => ordinary polymorphic default method -	-- Just True  => generic default method +        -- Nothing    => no default method +        -- Just False => ordinary polymorphic default method +        -- Just True  => generic default method  data IfaceConDecls -  = IfAbstractTyCon		-- No info -  | IfOpenDataTyCon		-- Open data family -  | IfDataTyCon [IfaceConDecl]	-- data type decls -  | IfNewTyCon  IfaceConDecl	-- newtype decls +  = IfAbstractTyCon             -- No info +  | IfOpenDataTyCon             -- Open data family +  | IfDataTyCon [IfaceConDecl]  -- data type decls +  | IfNewTyCon  IfaceConDecl    -- newtype decls  visibleIfConDecls :: IfaceConDecls -> [IfaceConDecl]  visibleIfConDecls IfAbstractTyCon  = [] @@ -123,49 +124,49 @@ visibleIfConDecls IfOpenDataTyCon  = []  visibleIfConDecls (IfDataTyCon cs) = cs  visibleIfConDecls (IfNewTyCon c)   = [c] -data IfaceConDecl  +data IfaceConDecl    = IfCon { -	ifConOcc     :: OccName,   		-- Constructor name -	ifConWrapper :: Bool,			-- True <=> has a wrapper -	ifConInfix   :: Bool,			-- True <=> declared infix -	ifConUnivTvs :: [IfaceTvBndr],		-- Universal tyvars -	ifConExTvs   :: [IfaceTvBndr],		-- Existential tyvars -	ifConEqSpec  :: [(OccName,IfaceType)],	-- Equality contraints -	ifConCtxt    :: IfaceContext,		-- Non-stupid context -	ifConArgTys  :: [IfaceType],		-- Arg types -	ifConFields  :: [OccName],		-- ...ditto... (field labels) -	ifConStricts :: [HsBang]}		-- Empty (meaning all lazy), -						-- or 1-1 corresp with arg tys - -data IfaceInst  -  = IfaceInst { ifInstCls  :: IfExtName,     		-- See comments with -		ifInstTys  :: [Maybe IfaceTyCon],	-- the defn of Instance -		ifDFun     :: IfExtName,     		-- The dfun -		ifOFlag    :: OverlapFlag,		-- Overlap flag -		ifInstOrph :: Maybe OccName }		-- See Note [Orphans] -	-- There's always a separate IfaceDecl for the DFun, which gives  -	-- its IdInfo with its full type and version number. -	-- The instance declarations taken together have a version number, -	-- and we don't want that to wobble gratuitously -	-- If this instance decl is *used*, we'll record a usage on the dfun; -	-- and if the head does not change it won't be used if it wasn't before +        ifConOcc     :: OccName,                -- Constructor name +        ifConWrapper :: Bool,                   -- True <=> has a wrapper +        ifConInfix   :: Bool,                   -- True <=> declared infix +        ifConUnivTvs :: [IfaceTvBndr],          -- Universal tyvars +        ifConExTvs   :: [IfaceTvBndr],          -- Existential tyvars +        ifConEqSpec  :: [(OccName,IfaceType)],  -- Equality contraints +        ifConCtxt    :: IfaceContext,           -- Non-stupid context +        ifConArgTys  :: [IfaceType],            -- Arg types +        ifConFields  :: [OccName],              -- ...ditto... (field labels) +        ifConStricts :: [HsBang]}               -- Empty (meaning all lazy), +                                                -- or 1-1 corresp with arg tys + +data IfaceInst +  = IfaceInst { ifInstCls  :: IfExtName,                -- See comments with +                ifInstTys  :: [Maybe IfaceTyCon],       -- the defn of Instance +                ifDFun     :: IfExtName,                -- The dfun +                ifOFlag    :: OverlapFlag,              -- Overlap flag +                ifInstOrph :: Maybe OccName }           -- See Note [Orphans] +        -- There's always a separate IfaceDecl for the DFun, which gives +        -- its IdInfo with its full type and version number. +        -- The instance declarations taken together have a version number, +        -- and we don't want that to wobble gratuitously +        -- If this instance decl is *used*, we'll record a usage on the dfun; +        -- and if the head does not change it won't be used if it wasn't before  data IfaceFamInst    = IfaceFamInst { ifFamInstFam   :: IfExtName                -- Family tycon -		 , ifFamInstTys   :: [Maybe IfaceTyCon]  -- Rough match types -		 , ifFamInstTyCon :: IfaceTyCon		 -- Instance decl -		 } +                 , ifFamInstTys   :: [Maybe IfaceTyCon]  -- Rough match types +                 , ifFamInstTyCon :: IfaceTyCon          -- Instance decl +                 }  data IfaceRule -  = IfaceRule {  -	ifRuleName   :: RuleName, -	ifActivation :: Activation, -	ifRuleBndrs  :: [IfaceBndr],	-- Tyvars and term vars -	ifRuleHead   :: IfExtName,   	-- Head of lhs -	ifRuleArgs   :: [IfaceExpr],	-- Args of LHS -	ifRuleRhs    :: IfaceExpr, -	ifRuleAuto   :: Bool, -	ifRuleOrph   :: Maybe OccName	-- Just like IfaceInst +  = IfaceRule { +        ifRuleName   :: RuleName, +        ifActivation :: Activation, +        ifRuleBndrs  :: [IfaceBndr],    -- Tyvars and term vars +        ifRuleHead   :: IfExtName,      -- Head of lhs +        ifRuleArgs   :: [IfaceExpr],    -- Args of LHS +        ifRuleRhs    :: IfaceExpr, +        ifRuleAuto   :: Bool, +        ifRuleOrph   :: Maybe OccName   -- Just like IfaceInst      }  data IfaceAnnotation @@ -187,51 +188,51 @@ data IfaceIdDetails    | IfDFunId Int          -- Number of silent args  data IfaceIdInfo -  = NoInfo			-- When writing interface file without -O -  | HasInfo [IfaceInfoItem]	-- Has info, and here it is +  = NoInfo                      -- When writing interface file without -O +  | HasInfo [IfaceInfoItem]     -- Has info, and here it is  -- Here's a tricky case:  --   * Compile with -O module A, and B which imports A.f  --   * Change function f in A, and recompile without -O  --   * When we read in old A.hi we read in its IdInfo (as a thunk) ---	(In earlier GHCs we used to drop IdInfo immediately on reading, ---	 but we do not do that now.  Instead it's discarded when the ---	 ModIface is read into the various decl pools.) +--      (In earlier GHCs we used to drop IdInfo immediately on reading, +--       but we do not do that now.  Instead it's discarded when the +--       ModIface is read into the various decl pools.)  --   * The version comparsion sees that new (=NoInfo) differs from old (=HasInfo *) ---	and so gives a new version. +--      and so gives a new version.  data IfaceInfoItem -  = HsArity	 Arity +  = HsArity      Arity    | HsStrictness StrictSig    | HsInline     InlinePragma -  | HsUnfold	 Bool		  -- True <=> isNonRuleLoopBreaker is true -		 IfaceUnfolding   -- See Note [Expose recursive functions]  +  | HsUnfold     Bool             -- True <=> isNonRuleLoopBreaker is true +                 IfaceUnfolding   -- See Note [Expose recursive functions]    | HsNoCafRefs  -- NB: Specialisations and rules come in separately and are  -- only later attached to the Id.  Partial reason: some are orphans. -data IfaceUnfolding  +data IfaceUnfolding    = IfCoreUnfold Bool IfaceExpr -- True <=> INLINABLE, False <=> regular unfolding                                  -- Possibly could eliminate the Bool here, the information                                  -- is also in the InlinePragma. -  | IfCompulsory IfaceExpr	-- Only used for default methods, in fact +  | IfCompulsory IfaceExpr      -- Only used for default methods, in fact    | IfInlineRule Arity          -- INLINE pragmas -                 Bool		-- OK to inline even if *un*-saturated -		 Bool		-- OK to inline even if context is boring -                 IfaceExpr  +                 Bool           -- OK to inline even if *un*-saturated +                 Bool           -- OK to inline even if context is boring +                 IfaceExpr -  | IfExtWrapper Arity IfExtName  -- NB: sometimes we need a IfExtName (not just IfLclName)  -  | IfLclWrapper Arity IfLclName  --     because the worker can simplify to a function in  -    		       		  --     another module. +  | IfExtWrapper Arity IfExtName  -- NB: sometimes we need a IfExtName (not just IfLclName) +  | IfLclWrapper Arity IfLclName  --     because the worker can simplify to a function in +                                  --     another module.    | IfDFunUnfold [DFunArg IfaceExpr]  --------------------------------  data IfaceExpr -  = IfaceLcl 	IfLclName +  = IfaceLcl    IfLclName    | IfaceExt    IfExtName    | IfaceType   IfaceType    | IfaceCo     IfaceType		-- We re-use IfaceType for coercions @@ -242,26 +243,26 @@ data IfaceExpr    | IfaceLet	IfaceBinding  IfaceExpr    | IfaceNote	IfaceNote IfaceExpr    | IfaceCast   IfaceExpr IfaceCoercion -  | IfaceLit	Literal -  | IfaceFCall	ForeignCall IfaceType +  | IfaceLit    Literal +  | IfaceFCall  ForeignCall IfaceType    | IfaceTick   Module Int  data IfaceNote = IfaceSCC CostCentre                 | IfaceCoreNote String  type IfaceAlt = (IfaceConAlt, [IfLclName], IfaceExpr) -	-- Note: IfLclName, not IfaceBndr (and same with the case binder) -	-- We reconstruct the kind/type of the thing from the context -	-- thus saving bulk in interface files +        -- Note: IfLclName, not IfaceBndr (and same with the case binder) +        -- We reconstruct the kind/type of the thing from the context +        -- thus saving bulk in interface files  data IfaceConAlt = IfaceDefault - 		 | IfaceDataAlt IfExtName -		 | IfaceTupleAlt Boxity -		 | IfaceLitAlt Literal +                 | IfaceDataAlt IfExtName +                 | IfaceTupleAlt Boxity +                 | IfaceLitAlt Literal  data IfaceBinding -  = IfaceNonRec	IfaceLetBndr IfaceExpr -  | IfaceRec 	[(IfaceLetBndr, IfaceExpr)] +  = IfaceNonRec IfaceLetBndr IfaceExpr +  | IfaceRec    [(IfaceLetBndr, IfaceExpr)]  -- IfaceLetBndr is like IfaceIdBndr, but has IdInfo too  -- It's used for *non-top-level* let/rec binders @@ -300,9 +301,9 @@ complicate the situation though. Consider  and suppose we are compiling module X:    module X where -	import M -	data T = ... -	instance C Int T where ... +        import M +        data T = ... +        instance C Int T where ...  This instance is an orphan, because when compiling a third module Y we  might get a constraint (C Int v), and we'd want to improve v to T.  So @@ -316,7 +317,7 @@ More precisely, an instance is an orphan iff    If there are fundeps, then for every fundep, at least one of the    names free in a *non-determined* part of the instance head is -  defined in this module.   +  defined in this module.  (Note that these conditions hold trivially if the class is locally  defined.) @@ -343,10 +344,10 @@ a functionally-dependent part of the instance decl.  E.g.  and suppose we are compiling module X:    module X where -	import M -	data S  = ... -	data T = ... -	instance C S T where ... +        import M +        data S  = ... +        data T = ... +        instance C S T where ...  If we base the instance verion on T, I'm worried that changing S to S'  would change T's version, but not S or S'.  But an importing module might @@ -357,8 +358,8 @@ and it seems deeply obscure, so I'm going to leave it for now.  Note [Versioning of rules]  ~~~~~~~~~~~~~~~~~~~~~~~~~~ -A rule that is not an orphan has an ifRuleOrph field of (Just n), where -n appears on the LHS of the rule; any change in the rule changes the version of n. +A rule that is not an orphan has an ifRuleOrph field of (Just n), where n +appears on the LHS of the rule; any change in the rule changes the version of n.  \begin{code} @@ -381,7 +382,7 @@ ifaceDeclSubBndrs IfaceData {ifCons = IfAbstractTyCon}  = []  ifaceDeclSubBndrs (IfaceData {ifName = tc_occ,                                ifCons = IfNewTyCon (                                          IfCon { ifConOcc = con_occ }), -                              ifFamInst = famInst})  +                              ifFamInst = famInst})    =   -- implicit coerion and (possibly) family instance coercion      (mkNewTyCoOcc tc_occ) : (famInstCo famInst tc_occ) ++        -- data constructor and worker (newtypes don't have a wrapper) @@ -389,8 +390,8 @@ ifaceDeclSubBndrs (IfaceData {ifName = tc_occ,  ifaceDeclSubBndrs (IfaceData {ifName = tc_occ, -			      ifCons = IfDataTyCon cons,  -			      ifFamInst = famInst}) +                              ifCons = IfDataTyCon cons, +                              ifFamInst = famInst})    =   -- (possibly) family instance coercion;        -- there is no implicit coercion for non-newtypes      famInstCo famInst tc_occ @@ -399,20 +400,20 @@ ifaceDeclSubBndrs (IfaceData {ifName = tc_occ,      ++ concatMap dc_occs cons    where      dc_occs con_decl -	| has_wrapper = [con_occ, work_occ, wrap_occ] -	| otherwise   = [con_occ, work_occ] -	where -	  con_occ  = ifConOcc con_decl			-- DataCon namespace -	  wrap_occ = mkDataConWrapperOcc con_occ	-- Id namespace -	  work_occ = mkDataConWorkerOcc con_occ		-- Id namespace -	  has_wrapper = ifConWrapper con_decl		-- This is the reason for -	  	      		     			-- having the ifConWrapper field! - -ifaceDeclSubBndrs (IfaceClass {ifCtxt = sc_ctxt, ifName = cls_occ,  -			       ifSigs = sigs, ifATs = ats }) +        | has_wrapper = [con_occ, work_occ, wrap_occ] +        | otherwise   = [con_occ, work_occ] +        where +          con_occ  = ifConOcc con_decl            -- DataCon namespace +          wrap_occ = mkDataConWrapperOcc con_occ  -- Id namespace +          work_occ = mkDataConWorkerOcc con_occ   -- Id namespace +          has_wrapper = ifConWrapper con_decl     -- This is the reason for +                                                  -- having the ifConWrapper field! + +ifaceDeclSubBndrs (IfaceClass {ifCtxt = sc_ctxt, ifName = cls_occ, +                               ifSigs = sigs, ifATs = ats })    = -- dictionary datatype:      --   type constructor -    tc_occ :  +    tc_occ :      --   (possibly) newtype coercion      co_occs ++      --    data constructor (DataCon namespace) @@ -429,14 +430,14 @@ ifaceDeclSubBndrs (IfaceClass {ifCtxt = sc_ctxt, ifName = cls_occ,      n_ctxt = length sc_ctxt      n_sigs = length sigs      tc_occ  = mkClassTyConOcc cls_occ -    dc_occ  = mkClassDataConOcc cls_occ	 +    dc_occ  = mkClassDataConOcc cls_occ      co_occs | is_newtype = [mkNewTyCoOcc tc_occ] -	    | otherwise  = [] +            | otherwise  = []      dcww_occ = mkDataConWorkerOcc dc_occ -    is_newtype = n_sigs + n_ctxt == 1			-- Sigh  +    is_newtype = n_sigs + n_ctxt == 1 -- Sigh  ifaceDeclSubBndrs (IfaceSyn {ifName = tc_occ, -			     ifFamInst = famInst}) +                             ifFamInst = famInst})    = famInstCo famInst tc_occ  ifaceDeclSubBndrs _ = [] @@ -452,46 +453,46 @@ instance Outputable IfaceDecl where    ppr = pprIfaceDecl  pprIfaceDecl :: IfaceDecl -> SDoc -pprIfaceDecl (IfaceId {ifName = var, ifType = ty,  +pprIfaceDecl (IfaceId {ifName = var, ifType = ty,                         ifIdDetails = details, ifIdInfo = info}) -  = sep [ ppr var <+> dcolon <+> ppr ty,  -    	  nest 2 (ppr details), -	  nest 2 (ppr info) ] +  = sep [ ppr var <+> dcolon <+> ppr ty, +          nest 2 (ppr details), +          nest 2 (ppr info) ]  pprIfaceDecl (IfaceForeign {ifName = tycon})    = hsep [ptext (sLit "foreign import type dotnet"), ppr tycon] -pprIfaceDecl (IfaceSyn {ifName = tycon, ifTyVars = tyvars,  -		        ifSynRhs = Just mono_ty,  +pprIfaceDecl (IfaceSyn {ifName = tycon, ifTyVars = tyvars, +                        ifSynRhs = Just mono_ty,                          ifFamInst = mbFamInst})    = hang (ptext (sLit "type") <+> pprIfaceDeclHead [] tycon tyvars)         4 (vcat [equals <+> ppr mono_ty, pprFamily mbFamInst]) -pprIfaceDecl (IfaceSyn {ifName = tycon, ifTyVars = tyvars,  -		        ifSynRhs = Nothing, ifSynKind = kind }) +pprIfaceDecl (IfaceSyn {ifName = tycon, ifTyVars = tyvars, +                        ifSynRhs = Nothing, ifSynKind = kind })    = hang (ptext (sLit "type family") <+> pprIfaceDeclHead [] tycon tyvars)         4 (dcolon <+> ppr kind)  pprIfaceDecl (IfaceData {ifName = tycon, ifGeneric = gen, ifCtxt = context, -			 ifTyVars = tyvars, ifCons = condecls,  -			 ifRec = isrec, ifFamInst = mbFamInst}) +                         ifTyVars = tyvars, ifCons = condecls, +                         ifRec = isrec, ifFamInst = mbFamInst})    = hang (pp_nd <+> pprIfaceDeclHead context tycon tyvars)         4 (vcat [pprRec isrec, pprGen gen, pp_condecls tycon condecls, -	        pprFamily mbFamInst]) +                pprFamily mbFamInst])    where      pp_nd = case condecls of -		IfAbstractTyCon -> ptext (sLit "data") -		IfOpenDataTyCon -> ptext (sLit "data family") -		IfDataTyCon _   -> ptext (sLit "data") -		IfNewTyCon _  	-> ptext (sLit "newtype") - -pprIfaceDecl (IfaceClass {ifCtxt = context, ifName = clas, ifTyVars = tyvars,  -			  ifFDs = fds, ifATs = ats, ifSigs = sigs,  -			  ifRec = isrec}) +                IfAbstractTyCon -> ptext (sLit "data") +                IfOpenDataTyCon -> ptext (sLit "data family") +                IfDataTyCon _   -> ptext (sLit "data") +                IfNewTyCon _    -> ptext (sLit "newtype") + +pprIfaceDecl (IfaceClass {ifCtxt = context, ifName = clas, ifTyVars = tyvars, +                          ifFDs = fds, ifATs = ats, ifSigs = sigs, +                          ifRec = isrec})    = hang (ptext (sLit "class") <+> pprIfaceDeclHead context clas tyvars <+> pprFundeps fds)         4 (vcat [pprRec isrec, -	        sep (map ppr ats), -		sep (map ppr sigs)]) +                sep (map ppr ats), +                sep (map ppr sigs)])  pprRec :: RecFlag -> SDoc  pprRec isrec = ptext (sLit "RecFlag") <+> ppr isrec @@ -509,68 +510,68 @@ instance Outputable IfaceClassOp where  pprIfaceDeclHead :: IfaceContext -> OccName -> [IfaceTvBndr] -> SDoc  pprIfaceDeclHead context thing tyvars -  = hsep [pprIfaceContext context, parenSymOcc thing (ppr thing),  -	  pprIfaceTvBndrs tyvars] +  = hsep [pprIfaceContext context, parenSymOcc thing (ppr thing), +          pprIfaceTvBndrs tyvars]  pp_condecls :: OccName -> IfaceConDecls -> SDoc  pp_condecls _  IfAbstractTyCon  = ptext (sLit "{- abstract -}")  pp_condecls tc (IfNewTyCon c)   = equals <+> pprIfaceConDecl tc c  pp_condecls _  IfOpenDataTyCon  = empty  pp_condecls tc (IfDataTyCon cs) = equals <+> sep (punctuate (ptext (sLit " |")) -							     (map (pprIfaceConDecl tc) cs)) +                                                            (map (pprIfaceConDecl tc) cs))  pprIfaceConDecl :: OccName -> IfaceConDecl -> SDoc  pprIfaceConDecl tc -	(IfCon { ifConOcc = name, ifConInfix = is_infix, ifConWrapper = has_wrap, -		 ifConUnivTvs = univ_tvs, ifConExTvs = ex_tvs,  -		 ifConEqSpec = eq_spec, ifConCtxt = ctxt, ifConArgTys = arg_tys,  -		 ifConStricts = strs, ifConFields = fields }) +        (IfCon { ifConOcc = name, ifConInfix = is_infix, ifConWrapper = has_wrap, +                 ifConUnivTvs = univ_tvs, ifConExTvs = ex_tvs, +                 ifConEqSpec = eq_spec, ifConCtxt = ctxt, ifConArgTys = arg_tys, +                 ifConStricts = strs, ifConFields = fields })    = sep [main_payload, -	 if is_infix then ptext (sLit "Infix") else empty, -	 if has_wrap then ptext (sLit "HasWrapper") else empty, -	 ppUnless (null strs) $ -	    nest 4 (ptext (sLit "Stricts:") <+> hsep (map ppr_bang strs)), -	 ppUnless (null fields) $ -	    nest 4 (ptext (sLit "Fields:") <+> hsep (map ppr fields))] +         if is_infix then ptext (sLit "Infix") else empty, +         if has_wrap then ptext (sLit "HasWrapper") else empty, +         ppUnless (null strs) $ +            nest 4 (ptext (sLit "Stricts:") <+> hsep (map ppr_bang strs)), +         ppUnless (null fields) $ +            nest 4 (ptext (sLit "Fields:") <+> hsep (map ppr fields))]    where -    ppr_bang HsNoBang = char '_'	-- Want to see these +    ppr_bang HsNoBang = char '_'        -- Want to see these      ppr_bang bang     = ppr bang -         -    main_payload = ppr name <+> dcolon <+>  -		   pprIfaceForAllPart (univ_tvs ++ ex_tvs) (eq_ctxt ++ ctxt) pp_tau -    eq_ctxt = [(IfaceEqPred (IfaceTyVar (occNameFS tv)) ty)  -	      | (tv,ty) <- eq_spec]  +    main_payload = ppr name <+> dcolon <+> +                   pprIfaceForAllPart (univ_tvs ++ ex_tvs) (eq_ctxt ++ ctxt) pp_tau -	-- A bit gruesome this, but we can't form the full con_tau, and ppr it, -	-- because we don't have a Name for the tycon, only an OccName +    eq_ctxt = [(IfaceEqPred (IfaceTyVar (occNameFS tv)) ty) +              | (tv,ty) <- eq_spec] + +        -- A bit gruesome this, but we can't form the full con_tau, and ppr it, +        -- because we don't have a Name for the tycon, only an OccName      pp_tau = case map pprParendIfaceType arg_tys ++ [pp_res_ty] of -		(t:ts) -> fsep (t : map (arrow <+>) ts) -		[]     -> panic "pp_con_taus" +                (t:ts) -> fsep (t : map (arrow <+>) ts) +                []     -> panic "pp_con_taus"      pp_res_ty = ppr tc <+> fsep [ppr tv | (tv,_) <- univ_tvs]  instance Outputable IfaceRule where    ppr (IfaceRule { ifRuleName = name, ifActivation = act, ifRuleBndrs = bndrs, -		   ifRuleHead = fn, ifRuleArgs = args, ifRuleRhs = rhs })  +                   ifRuleHead = fn, ifRuleArgs = args, ifRuleRhs = rhs })      = sep [hsep [doubleQuotes (ftext name), ppr act, -		 ptext (sLit "forall") <+> pprIfaceBndrs bndrs], -	   nest 2 (sep [ppr fn <+> sep (map pprParendIfaceExpr args), -		        ptext (sLit "=") <+> ppr rhs]) +                 ptext (sLit "forall") <+> pprIfaceBndrs bndrs], +           nest 2 (sep [ppr fn <+> sep (map pprParendIfaceExpr args), +                        ptext (sLit "=") <+> ppr rhs])        ]  instance Outputable IfaceInst where -  ppr (IfaceInst {ifDFun = dfun_id, ifOFlag = flag,  -		  ifInstCls = cls, ifInstTys = mb_tcs}) -    = hang (ptext (sLit "instance") <+> ppr flag  -		<+> ppr cls <+> brackets (pprWithCommas ppr_rough mb_tcs)) +  ppr (IfaceInst {ifDFun = dfun_id, ifOFlag = flag, +                  ifInstCls = cls, ifInstTys = mb_tcs}) +    = hang (ptext (sLit "instance") <+> ppr flag +                <+> ppr cls <+> brackets (pprWithCommas ppr_rough mb_tcs))           2 (equals <+> ppr dfun_id)  instance Outputable IfaceFamInst where    ppr (IfaceFamInst {ifFamInstFam = fam, ifFamInstTys = mb_tcs, -		     ifFamInstTyCon = tycon_id}) -    = hang (ptext (sLit "family instance") <+>  -	    ppr fam <+> brackets (pprWithCommas ppr_rough mb_tcs)) +                     ifFamInstTyCon = tycon_id}) +    = hang (ptext (sLit "family instance") <+> +            ppr fam <+> brackets (pprWithCommas ppr_rough mb_tcs))           2 (equals <+> ppr tycon_id)  ppr_rough :: Maybe IfaceTyCon -> SDoc @@ -588,9 +589,11 @@ instance Outputable IfaceExpr where  pprParendIfaceExpr :: IfaceExpr -> SDoc  pprParendIfaceExpr = pprIfaceExpr parens +-- | Pretty Print an IfaceExpre +-- +-- The first argument should be a function that adds parens in context that need +-- an atomic value (e.g. function args)  pprIfaceExpr :: (SDoc -> SDoc) -> IfaceExpr -> SDoc -	-- The function adds parens in context that need -	-- an atomic value (e.g. function args)  pprIfaceExpr _       (IfaceLcl v)       = ppr v  pprIfaceExpr _       (IfaceExt v)       = ppr v @@ -603,11 +606,11 @@ pprIfaceExpr _       (IfaceCo co)       = text "@~" <+> pprParendIfaceType co  pprIfaceExpr add_par app@(IfaceApp _ _) = add_par (pprIfaceApp app [])  pprIfaceExpr _       (IfaceTuple c as)  = tupleParens c (interpp'SP as) -pprIfaceExpr add_par e@(IfaceLam _ _)    +pprIfaceExpr add_par i@(IfaceLam _ _)    = add_par (sep [char '\\' <+> sep (map ppr bndrs) <+> arrow, -		  pprIfaceExpr noParens body]) -  where  -    (bndrs,body) = collect [] e +                  pprIfaceExpr noParens body]) +  where +    (bndrs,body) = collect [] i      collect bs (IfaceLam b e) = collect (b:bs) e      collect bs e              = (reverse bs, e) @@ -625,78 +628,85 @@ pprIfaceExpr add_par (IfaceCase scrut bndr alts)  pprIfaceExpr _       (IfaceCast expr co)    = sep [pprParendIfaceExpr expr, -	 nest 2 (ptext (sLit "`cast`")), -	 pprParendIfaceType co] +         nest 2 (ptext (sLit "`cast`")), +         pprParendIfaceType co]  pprIfaceExpr add_par (IfaceLet (IfaceNonRec b rhs) body) -  = add_par (sep [ptext (sLit "let {"),  -		  nest 2 (ppr_bind (b, rhs)), -		  ptext (sLit "} in"),  -		  pprIfaceExpr noParens body]) +  = add_par (sep [ptext (sLit "let {"), +                  nest 2 (ppr_bind (b, rhs)), +                  ptext (sLit "} in"), +                  pprIfaceExpr noParens body])  pprIfaceExpr add_par (IfaceLet (IfaceRec pairs) body)    = add_par (sep [ptext (sLit "letrec {"), -		  nest 2 (sep (map ppr_bind pairs)),  -		  ptext (sLit "} in"), -		  pprIfaceExpr noParens body]) +                  nest 2 (sep (map ppr_bind pairs)), +                  ptext (sLit "} in"), +                  pprIfaceExpr noParens body]) -pprIfaceExpr add_par (IfaceNote note body) = add_par (ppr note <+> pprParendIfaceExpr body) +pprIfaceExpr add_par (IfaceNote note body) = add_par $ ppr note +                                                <+> pprParendIfaceExpr body  ppr_alt :: (IfaceConAlt, [IfLclName], IfaceExpr) -> SDoc -ppr_alt (con, bs, rhs) = sep [ppr_con_bs con bs,  -			      arrow <+> pprIfaceExpr noParens rhs] +ppr_alt (con, bs, rhs) = sep [ppr_con_bs con bs, +                         arrow <+> pprIfaceExpr noParens rhs]  ppr_con_bs :: IfaceConAlt -> [IfLclName] -> SDoc  ppr_con_bs (IfaceTupleAlt tup_con) bs = tupleParens tup_con (interpp'SP bs) -ppr_con_bs con bs		      = ppr con <+> hsep (map ppr bs) -   +ppr_con_bs con bs                     = ppr con <+> hsep (map ppr bs) +  ppr_bind :: (IfaceLetBndr, IfaceExpr) -> SDoc -ppr_bind (IfLetBndr b ty info, rhs)  +ppr_bind (IfLetBndr b ty info, rhs)    = sep [hang (ppr b <+> dcolon <+> ppr ty) 2 (ppr info), -	 equals <+> pprIfaceExpr noParens rhs] +         equals <+> pprIfaceExpr noParens rhs]  ------------------  pprIfaceApp :: IfaceExpr -> [SDoc] -> SDoc -pprIfaceApp (IfaceApp fun arg) args = pprIfaceApp fun (nest 2 (pprParendIfaceExpr arg) : args) -pprIfaceApp fun	 	       args = sep (pprParendIfaceExpr fun : args) +pprIfaceApp (IfaceApp fun arg) args = pprIfaceApp fun $ +                                          nest 2 (pprParendIfaceExpr arg) : args +pprIfaceApp fun                args = sep (pprParendIfaceExpr fun : args)  ------------------  instance Outputable IfaceNote where      ppr (IfaceSCC cc)     = pprCostCentreCore cc -    ppr (IfaceCoreNote s) = ptext (sLit "__core_note") <+> pprHsString (mkFastString s) +    ppr (IfaceCoreNote s) = ptext (sLit "__core_note") +                            <+> pprHsString (mkFastString s)  instance Outputable IfaceConAlt where      ppr IfaceDefault      = text "DEFAULT"      ppr (IfaceLitAlt l)   = ppr l      ppr (IfaceDataAlt d)  = ppr d -    ppr (IfaceTupleAlt _) = panic "ppr IfaceConAlt"  +    ppr (IfaceTupleAlt _) = panic "ppr IfaceConAlt"      -- IfaceTupleAlt is handled by the case-alternative printer  ------------------  instance Outputable IfaceIdDetails where -  ppr IfVanillaId    = empty +  ppr IfVanillaId       = empty    ppr (IfRecSelId tc b) = ptext (sLit "RecSel") <+> ppr tc -      		          <+> if b then ptext (sLit "<naughty>") else empty +                          <+> if b then ptext (sLit "<naughty>") else empty    ppr (IfDFunId ns)     = ptext (sLit "DFunId") <> brackets (int ns)  instance Outputable IfaceIdInfo where    ppr NoInfo       = empty -  ppr (HasInfo is) = ptext (sLit "{-") <+> pprWithCommas ppr is <+> ptext (sLit "-}") +  ppr (HasInfo is) = ptext (sLit "{-") <+> pprWithCommas ppr is +                     <+> ptext (sLit "-}")  instance Outputable IfaceInfoItem where -  ppr (HsUnfold lb unf)  = ptext (sLit "Unfolding") <> ppWhen lb (ptext (sLit "(loop-breaker)"))  +  ppr (HsUnfold lb unf)  = ptext (sLit "Unfolding") +                           <> ppWhen lb (ptext (sLit "(loop-breaker)"))                             <> colon <+> ppr unf    ppr (HsInline prag)    = ptext (sLit "Inline:") <+> ppr prag    ppr (HsArity arity)    = ptext (sLit "Arity:") <+> int arity    ppr (HsStrictness str) = ptext (sLit "Strictness:") <+> pprIfaceStrictSig str -  ppr HsNoCafRefs	 = ptext (sLit "HasNoCafRefs") +  ppr HsNoCafRefs        = ptext (sLit "HasNoCafRefs")  instance Outputable IfaceUnfolding where    ppr (IfCompulsory e)     = ptext (sLit "<compulsory>") <+> parens (ppr e) -  ppr (IfCoreUnfold s e)   = (if s then ptext (sLit "<stable>") else empty) <+> parens (ppr e) -  ppr (IfInlineRule a uok bok e) = sep [ptext (sLit "InlineRule") <+> ppr (a,uok,bok), -      		           	        pprParendIfaceExpr e] +  ppr (IfCoreUnfold s e)   = (if s then ptext (sLit "<stable>") else empty) +                              <+> parens (ppr e) +  ppr (IfInlineRule a uok bok e) = sep [ptext (sLit "InlineRule") +                                            <+> ppr (a,uok,bok), +                                        pprParendIfaceExpr e]    ppr (IfLclWrapper a wkr) = ptext (sLit "Worker(lcl):") <+> ppr wkr                               <+> parens (ptext (sLit "arity") <+> int a)    ppr (IfExtWrapper a wkr) = ptext (sLit "Worker(ext0:") <+> ppr wkr @@ -705,7 +715,7 @@ instance Outputable IfaceUnfolding where                               <+> brackets (pprWithCommas ppr ns)  -- ----------------------------------------------------------------------------- --- Finding the Names in IfaceSyn +-- | Finding the Names in IfaceSyn  -- This is used for dependency analysis in MkIface, so that we  -- fingerprint a declaration before the things that depend on it.  It @@ -715,11 +725,11 @@ instance Outputable IfaceUnfolding where  -- fingerprinting the instance, so DFuns are not dependencies.  freeNamesIfDecl :: IfaceDecl -> NameSet -freeNamesIfDecl (IfaceId _s t d i) =  +freeNamesIfDecl (IfaceId _s t d i) =    freeNamesIfType t &&&    freeNamesIfIdInfo i &&&    freeNamesIfIdDetails d -freeNamesIfDecl IfaceForeign{} =  +freeNamesIfDecl IfaceForeign{} =    emptyNameSet  freeNamesIfDecl d@IfaceData{} =    freeNamesIfTvBndrs (ifTyVars d) &&& @@ -746,7 +756,7 @@ freeNamesIfSynRhs (Just ty) = freeNamesIfType ty  freeNamesIfSynRhs Nothing   = emptyNameSet  freeNamesIfTcFam :: Maybe (IfaceTyCon, [IfaceType]) -> NameSet -freeNamesIfTcFam (Just (tc,tys)) =  +freeNamesIfTcFam (Just (tc,tys)) =    freeNamesIfTc tc &&& fnList freeNamesIfType tys  freeNamesIfTcFam Nothing =    emptyNameSet @@ -766,15 +776,15 @@ freeNamesIfConDecls (IfNewTyCon c)  = freeNamesIfConDecl c  freeNamesIfConDecls _               = emptyNameSet  freeNamesIfConDecl :: IfaceConDecl -> NameSet -freeNamesIfConDecl c =  +freeNamesIfConDecl c =    freeNamesIfTvBndrs (ifConUnivTvs c) &&&    freeNamesIfTvBndrs (ifConExTvs c) &&& -  freeNamesIfContext (ifConCtxt c) &&&  +  freeNamesIfContext (ifConCtxt c) &&&    fnList freeNamesIfType (ifConArgTys c) &&&    fnList freeNamesIfType (map snd (ifConEqSpec c)) -- equality constraints  freeNamesIfPredType :: IfacePredType -> NameSet -freeNamesIfPredType (IfaceClassP cl tys) =  +freeNamesIfPredType (IfaceClassP cl tys) =     unitNameSet cl &&& fnList freeNamesIfType tys  freeNamesIfPredType (IfaceIParam _n ty) =     freeNamesIfType ty @@ -785,7 +795,7 @@ freeNamesIfType :: IfaceType -> NameSet  freeNamesIfType (IfaceTyVar _)        = emptyNameSet  freeNamesIfType (IfaceAppTy s t)      = freeNamesIfType s &&& freeNamesIfType t  freeNamesIfType (IfacePredTy st)      = freeNamesIfPredType st -freeNamesIfType (IfaceTyConApp tc ts) =  +freeNamesIfType (IfaceTyConApp tc ts) =     freeNamesIfTc tc &&& fnList freeNamesIfType ts  freeNamesIfType (IfaceForAllTy tv t)  =     freeNamesIfTvBndr tv &&& freeNamesIfType t @@ -802,7 +812,7 @@ freeNamesIfBndr (IfaceTvBndr b) = freeNamesIfTvBndr b  freeNamesIfLetBndr :: IfaceLetBndr -> NameSet  -- Remember IfaceLetBndr is used only for *nested* bindings --- The IdInfo can contain an unfolding (in the case of  +-- The IdInfo can contain an unfolding (in the case of  -- local INLINE pragmas), so look there too  freeNamesIfLetBndr (IfLetBndr _name ty info) = freeNamesIfType ty                                               &&& freeNamesIfIdInfo info @@ -815,7 +825,7 @@ freeNamesIfIdBndr :: IfaceIdBndr -> NameSet  freeNamesIfIdBndr = freeNamesIfTvBndr  freeNamesIfIdInfo :: IfaceIdInfo -> NameSet -freeNamesIfIdInfo NoInfo = emptyNameSet +freeNamesIfIdInfo NoInfo      = emptyNameSet  freeNamesIfIdInfo (HasInfo i) = fnList freeNamesItem i  freeNamesItem :: IfaceInfoItem -> NameSet @@ -831,7 +841,7 @@ freeNamesIfUnfold (IfLclWrapper {})      = emptyNameSet  freeNamesIfUnfold (IfDFunUnfold vs)      = fnList freeNamesIfExpr (dfunArgExprs vs)  freeNamesIfExpr :: IfaceExpr -> NameSet -freeNamesIfExpr (IfaceExt v)	  = unitNameSet v +freeNamesIfExpr (IfaceExt v)      = unitNameSet v  freeNamesIfExpr (IfaceFCall _ ty) = freeNamesIfType ty  freeNamesIfExpr (IfaceType ty)    = freeNamesIfType ty  freeNamesIfExpr (IfaceCo co)      = freeNamesIfType co @@ -839,7 +849,7 @@ freeNamesIfExpr (IfaceTuple _ as) = fnList freeNamesIfExpr as  freeNamesIfExpr (IfaceLam b body) = freeNamesIfBndr b &&& freeNamesIfExpr body  freeNamesIfExpr (IfaceApp f a)    = freeNamesIfExpr f &&& freeNamesIfExpr a  freeNamesIfExpr (IfaceCast e co)  = freeNamesIfExpr e &&& freeNamesIfType co -freeNamesIfExpr (IfaceNote _n r)   = freeNamesIfExpr r +freeNamesIfExpr (IfaceNote _n r)  = freeNamesIfExpr r  freeNamesIfExpr (IfaceCase s _ alts)    = freeNamesIfExpr s  @@ -849,10 +859,10 @@ freeNamesIfExpr (IfaceCase s _ alts)      -- Depend on the data constructors.  Just one will do!      -- Note [Tracking data constructors] -    fn_cons []                              = emptyNameSet -    fn_cons ((IfaceDefault    ,_,_) : alts) = fn_cons alts -    fn_cons ((IfaceDataAlt con,_,_) : _   ) = unitNameSet con     -    fn_cons (_                      : _   ) = emptyNameSet +    fn_cons []                            = emptyNameSet +    fn_cons ((IfaceDefault    ,_,_) : xs) = fn_cons xs +    fn_cons ((IfaceDataAlt con,_,_) : _ ) = unitNameSet con +    fn_cons (_                      : _ ) = emptyNameSet  freeNamesIfExpr (IfaceLet (IfaceNonRec bndr rhs) body)    = freeNamesIfLetBndr bndr &&& freeNamesIfExpr rhs &&& freeNamesIfExpr body @@ -891,18 +901,18 @@ fnList f = foldr (&&&) emptyNameSet . map f  Note [Tracking data constructors]  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -In a case expression  +In a case expression     case e of { C a -> ...; ... }  You might think that we don't need to include the datacon C -in the free names, because its type will probably show up in  +in the free names, because its type will probably show up in  the free names of 'e'.  But in rare circumstances this may  not happen.   Here's the one that bit me: -   module DynFlags where  +   module DynFlags where       import {-# SOURCE #-} Packages( PackageState )       data DynFlags = DF ... PackageState ... -   module Packages where  +   module Packages where       import DynFlags       data PackageState = PS ...       lookupModule (df :: DynFlags) @@ -913,3 +923,4 @@ not happen.   Here's the one that bit me:  Now, lookupModule depends on DynFlags, but the transitive dependency  on the *locally-defined* type PackageState is not visible. We need  to take account of the use of the data constructor PS in the pattern match. + diff --git a/compiler/llvmGen/LlvmCodeGen/Ppr.hs b/compiler/llvmGen/LlvmCodeGen/Ppr.hs index 911592bc20..9f25c08826 100644 --- a/compiler/llvmGen/LlvmCodeGen/Ppr.hs +++ b/compiler/llvmGen/LlvmCodeGen/Ppr.hs @@ -122,34 +122,25 @@ pprInfoTable env count lbl stat            then Outputable.panic "LlvmCodeGen.Ppr: invalid info table!"            else (pprLlvmData ([ldata'], ltypes), llvmUsed) +  -- | We generate labels for info tables by converting them to the same label  -- as for the entry code but adding this string as a suffix.  iTableSuf :: String  iTableSuf = "_itable" --- | Create an appropriate section declaration for subsection <n> of text --- WARNING: This technique could fail as gas documentation says it only --- supports up to 8192 subsections per section. Inspection of the source --- code and some test programs seem to suggest it supports more than this --- so we are hoping it does. +-- | Create a specially crafted section declaration that encodes the order this +-- section should be in the final object code. +--  +-- The LlvmMangler.llvmFixupAsm pass over the assembly produced by LLVM uses +-- this section declaration to do its processing.  mkLayoutSection :: Int -> LMSection  mkLayoutSection n -  -- On OSX we can't use the GNU Assembler, we must use the OSX assembler, which -  -- doesn't support subsections. So we post process the assembly code, this -  -- section specifier will be replaced with '.text' by the mangler. -  = Just (fsLit $ infoSection ++ show n -#if darwin_TARGET_OS -      ) -#else -      ++ "#") -#endif +  = Just (fsLit $ infoSection ++ show n) --- | The section we are putting info tables and their entry code into + +-- | The section we are putting info tables and their entry code into, should +-- be unique since we process the assembly pattern matching this.  infoSection :: String -#if darwin_TARGET_OS -infoSection = "__STRIP,__me" -#else -infoSection = ".text; .text " -#endif +infoSection = "X98A__STRIP,__me" diff --git a/compiler/llvmGen/LlvmMangler.hs b/compiler/llvmGen/LlvmMangler.hs index 7b38ed8fa2..591ef81934 100644 --- a/compiler/llvmGen/LlvmMangler.hs +++ b/compiler/llvmGen/LlvmMangler.hs @@ -1,17 +1,21 @@ +{-# OPTIONS -fno-warn-unused-binds #-}  -- -----------------------------------------------------------------------------  -- | GHC LLVM Mangler  --  -- This script processes the assembly produced by LLVM, rearranging the code --- so that an info table appears before its corresponding function. We also --- use it to fix up the stack alignment, which needs to be 16 byte aligned --- but always ends up off by 4 bytes because GHC sets it to the 'wrong' --- starting value in the RTS. +-- so that an info table appears before its corresponding function.  -- --- We only need this for Mac OS X, other targets don't use it. +-- On OSX we also use it to fix up the stack alignment, which needs to be 16 +-- byte aligned but always ends up off by word bytes because GHC sets it to +-- the 'wrong' starting value in the RTS.  --  module LlvmMangler ( llvmFixupAsm ) where +#include "HsVersions.h" + +import LlvmCodeGen.Ppr ( infoSection ) +  import Control.Exception  import qualified Data.ByteString.Char8 as B  import Data.Char @@ -19,17 +23,24 @@ import qualified Data.IntMap as I  import System.IO  -- Magic Strings -infoSec, newInfoSec, newLine, spInst, jmpInst :: B.ByteString -infoSec    = B.pack "\t.section\t__STRIP,__me" +secStmt, infoSec, newInfoSec, newLine, spInst, jmpInst :: B.ByteString +secStmt    = B.pack "\t.section\t" +infoSec    = B.pack infoSection  newInfoSec = B.pack "\n\t.text"  newLine    = B.pack "\n" -spInst     = B.pack ", %esp\n"  jmpInst    = B.pack "\n\tjmp" -infoLen, spFix, labelStart :: Int -infoLen = B.length infoSec -spFix   = 4 -labelStart = B.length jmpInst + 1 +infoLen, labelStart, spFix :: Int +infoLen    = B.length infoSec +labelStart = B.length jmpInst + +#if x86_64_TARGET_ARCH +spInst     = B.pack ", %rsp\n" +spFix      = 8 +#else +spInst     = B.pack ", %esp\n" +spFix      = 4 +#endif  -- Search Predicates  eolPred, dollarPred, commaPred :: Char -> Bool @@ -50,25 +61,30 @@ llvmFixupAsm f1 f2 = do  {- |      Here we process the assembly file one function and data -    defenition at a time. When a function is encountered that +    definition at a time. When a function is encountered that      should have a info table we store it in a map. Otherwise      we print it. When an info table is found we retrieve its      function from the map and print them both.      For all functions we fix up the stack alignment. We also -    fix up the section defenition for functions and info tables. +    fix up the section definition for functions and info tables.  -}  fixTables :: Handle -> Handle -> I.IntMap B.ByteString -> IO ()  fixTables r w m = do      f <- getFun r B.empty      if B.null f         then return () -       else let fun   = fixupStack f B.empty -                (a,b) = B.breakSubstring infoSec fun -                (x,c) = B.break eolPred b -                fun'  = a `B.append` newInfoSec `B.append` c -                n     = readInt $ B.drop infoLen x -                (bs, m') | B.null b  = ([fun], m) +       else let fun    = fixupStack f B.empty +                (a,b)  = B.breakSubstring infoSec fun +                (a',s) = B.breakEnd eolPred a +                -- We search for the section header in two parts as it makes +                -- us portable across OS types and LLVM version types since +                -- section names are wrapped differently. +                secHdr = secStmt `B.isPrefixOf` s +                (x,c)  = B.break eolPred b +                fun'   = a' `B.append` newInfoSec `B.append` c +                n      = readInt $ B.takeWhile isDigit $ B.drop infoLen x +                (bs, m') | B.null b || not secHdr = ([fun], m)                           | even n    = ([], I.insert n fun' m)                           | otherwise = case I.lookup (n+1) m of                                 Just xf' -> ([fun',xf'], m) @@ -88,7 +104,7 @@ getFun r f = do      Mac OS X requires that the stack be 16 byte aligned when making a function      call (only really required though when making a call that will pass through      the dynamic linker). The alignment isn't correctly generated by LLVM as -    LLVM rightly assumes that the stack wil be aligned to 16n + 12 on entry +    LLVM rightly assumes that the stack will be aligned to 16n + 12 on entry      (since the function call was 16 byte aligned and the return address should      have been pushed, so sub 4). GHC though since it always uses jumps keeps      the stack 16 byte aligned on both function calls and function entry. @@ -96,6 +112,11 @@ getFun r f = do      We correct the alignment here.  -}  fixupStack :: B.ByteString -> B.ByteString -> B.ByteString + +#if !darwin_TARGET_OS +fixupStack = const + +#else  fixupStack f f' | B.null f' =      let -- fixup sub op          (a, c) = B.breakSubstring spInst f @@ -114,18 +135,21 @@ fixupStack f f' =          (a', n) = B.breakEnd dollarPred a          (n', x) = B.break commaPred n          num     = B.pack $ show $ readInt n' + spFix +        -- We need to avoid processing jumps to labels, they are of the form: +        -- jmp\tL..., jmp\t_f..., jmpl\t_f..., jmpl\t*%eax..., jmpl *L... +        targ = B.dropWhile ((==)'*') $ B.drop 1 $ B.dropWhile ((/=)'\t') $ +                B.drop labelStart c      in if B.null c            then f' `B.append` f -          -- We need to avoid processing jumps to labels, they are of the form: -          -- jmp\tL..., jmp\t_f..., jmpl\t_f..., jmpl\t*%eax... -          else if B.index c labelStart == 'L' +          else if B.head targ == 'L'                  then fixupStack b $ f' `B.append` a `B.append` l                  else fixupStack b $ f' `B.append` a' `B.append` num `B.append`                                      x `B.append` l +#endif --- | read an int or error +-- | Read an int or error  readInt :: B.ByteString -> Int  readInt str | B.all isDigit str = (read . B.unpack) str -            | otherwise = error $ "LLvmMangler Cannot read" ++ show str -                                ++ "as it's not an Int" +            | otherwise = error $ "LLvmMangler Cannot read " ++ show str +                                ++ " as it's not an Int" diff --git a/compiler/main/CodeOutput.lhs b/compiler/main/CodeOutput.lhs index f5030777cb..f5e339440b 100644 --- a/compiler/main/CodeOutput.lhs +++ b/compiler/main/CodeOutput.lhs @@ -8,9 +8,7 @@ module CodeOutput( codeOutput, outputForeignStubs ) where  #include "HsVersions.h" -#ifndef OMIT_NATIVE_CODEGEN -import AsmCodeGen	( nativeCodeGen ) -#endif +import AsmCodeGen ( nativeCodeGen )  import LlvmCodeGen ( llvmCodeGen )  import UniqSupply	( mkSplitUniqSupply ) @@ -149,24 +147,16 @@ outputC dflags filenm flat_absC packages  \begin{code}  outputAsm :: DynFlags -> FilePath -> [RawCmm] -> IO () - -#ifndef OMIT_NATIVE_CODEGEN -  outputAsm dflags filenm flat_absC + | cGhcWithNativeCodeGen == "YES"    = do ncg_uniqs <- mkSplitUniqSupply 'n'         {-# SCC "OutputAsm" #-} doOutput filenm $ -	   \f -> {-# SCC "NativeCodeGen" #-} -	         nativeCodeGen dflags f ncg_uniqs flat_absC -  where +           \f -> {-# SCC "NativeCodeGen" #-} +                 nativeCodeGen dflags f ncg_uniqs flat_absC -#else /* OMIT_NATIVE_CODEGEN */ - -outputAsm _ _ _ -  = pprPanic "This compiler was built without a native code generator" -	     (text "Use -fvia-C instead") - -#endif + | otherwise +  = panic "This compiler was built without a native code generator"  \end{code} diff --git a/compiler/main/DriverPhases.hs b/compiler/main/DriverPhases.hs index f6a9738af1..4702682ee4 100644 --- a/compiler/main/DriverPhases.hs +++ b/compiler/main/DriverPhases.hs @@ -143,11 +143,7 @@ nextPhase (Hsc   _)     = HCc  nextPhase SplitMangle   = As  nextPhase As            = SplitAs  nextPhase LlvmOpt       = LlvmLlc -#if darwin_TARGET_OS  nextPhase LlvmLlc       = LlvmMangle -#else -nextPhase LlvmLlc       = As -#endif  nextPhase LlvmMangle    = As  nextPhase SplitAs       = MergeStub  nextPhase Ccpp          = As diff --git a/compiler/main/DriverPipeline.hs b/compiler/main/DriverPipeline.hs index 70d99d40af..a832034749 100644 --- a/compiler/main/DriverPipeline.hs +++ b/compiler/main/DriverPipeline.hs @@ -55,6 +55,7 @@ import MonadUtils  -- import Data.Either  import Exception  import Data.IORef       ( readIORef ) +import Distribution.System  -- import GHC.Exts              ( Int(..) )  import System.Directory  import System.FilePath @@ -269,11 +270,11 @@ link :: GhcLink                 -- interactive or batch  -- exports main, i.e., we have good reason to believe that linking  -- will succeed. -#ifdef GHCI  link LinkInMemory _ _ _ -    = do -- Not Linking...(demand linker will do the job) -         return Succeeded -#endif +    = if cGhcWithInterpreter == "YES" +      then -- Not Linking...(demand linker will do the job) +           return Succeeded +      else panicBadLink LinkInMemory  link NoLink _ _ _     = return Succeeded @@ -284,11 +285,6 @@ link LinkBinary dflags batch_attempt_linking hpt  link LinkDynLib dflags batch_attempt_linking hpt     = link' dflags batch_attempt_linking hpt -#ifndef GHCI --- warning suppression -link other _ _ _ = panicBadLink other -#endif -  panicBadLink :: GhcLink -> a  panicBadLink other = panic ("link: GHC not built to link this way: " ++                              show other) @@ -1061,15 +1057,14 @@ runPhase cc_phase input_fn dflags          let            more_hcc_opts = -#if i386_TARGET_ARCH                  -- on x86 the floating point regs have greater precision                  -- than a double, which leads to unpredictable results.                  -- By default, we turn this off with -ffloat-store unless                  -- the user specified -fexcess-precision. -                (if dopt Opt_ExcessPrecision dflags -                        then [] -                        else [ "-ffloat-store" ]) ++ -#endif +                (if cTargetArch == I386 && +                    not (dopt Opt_ExcessPrecision dflags) +                        then [ "-ffloat-store" ] +                        else []) ++                  -- gcc's -fstrict-aliasing allows two accesses to memory                  -- to be considered non-aliasing if they have different types. @@ -1093,26 +1088,26 @@ runPhase cc_phase input_fn dflags                         ++ map SysTools.Option (                            pic_c_flags -#if    defined(mingw32_TARGET_OS)                  -- Stub files generated for foreign exports references the runIO_closure                  -- and runNonIO_closure symbols, which are defined in the base package.                  -- These symbols are imported into the stub.c file via RtsAPI.h, and the                  -- way we do the import depends on whether we're currently compiling                  -- the base package or not. -                       ++ (if thisPackage dflags == basePackageId +                       ++ (if cTargetOS == Windows && +                              thisPackage dflags == basePackageId                                  then [ "-DCOMPILING_BASE_PACKAGE" ]                                  else []) -#endif -#ifdef sparc_TARGET_ARCH          -- We only support SparcV9 and better because V8 lacks an atomic CAS          -- instruction. Note that the user can still override this          -- (e.g., -mcpu=ultrasparc) as GCC picks the "best" -mcpu flag          -- regardless of the ordering.          --          -- This is a temporary hack. -                       ++ ["-mcpu=v9"] -#endif +                       ++ (if cTargetArch == Sparc +                           then ["-mcpu=v9"] +                           else []) +                         ++ (if hcc                               then gcc_extra_viac_flags ++ more_hcc_opts                               else []) @@ -1179,7 +1174,7 @@ runPhase As input_fn dflags          io $ SysTools.runAs dflags                         (map SysTools.Option as_opts                         ++ [ SysTools.Option ("-I" ++ p) | p <- cmdline_include_paths ] -#ifdef sparc_TARGET_ARCH +          -- We only support SparcV9 and better because V8 lacks an atomic CAS          -- instruction so we have to make sure that the assembler accepts the          -- instruction set. Note that the user can still override this @@ -1187,8 +1182,10 @@ runPhase As input_fn dflags          -- regardless of the ordering.          --          -- This is a temporary hack. -                       ++ [ SysTools.Option "-mcpu=v9" ] -#endif +                       ++ (if cTargetArch == Sparc +                           then [SysTools.Option "-mcpu=v9"] +                           else []) +                         ++ [ SysTools.Option "-c"                            , SysTools.FileOption "" input_fn                            , SysTools.Option "-o" @@ -1232,7 +1229,7 @@ runPhase SplitAs _input_fn dflags          let assemble_file n                = SysTools.runAs dflags                           (map SysTools.Option as_opts ++ -#ifdef sparc_TARGET_ARCH +          -- We only support SparcV9 and better because V8 lacks an atomic CAS          -- instruction so we have to make sure that the assembler accepts the          -- instruction set. Note that the user can still override this @@ -1240,8 +1237,10 @@ runPhase SplitAs _input_fn dflags          -- regardless of the ordering.          --          -- This is a temporary hack. -                          [ SysTools.Option "-mcpu=v9" ] ++ -#endif +                          (if cTargetArch == Sparc +                           then [SysTools.Option "-mcpu=v9"] +                           else []) ++ +                            [ SysTools.Option "-c"                            , SysTools.Option "-o"                            , SysTools.FileOption "" (split_obj n) @@ -1308,24 +1307,18 @@ runPhase LlvmOpt input_fn dflags          -- fix up some pretty big deficiencies in the code we generate          llvmOpts = ["-mem2reg", "-O1", "-O2"] -  -----------------------------------------------------------------------------  -- LlvmLlc phase  runPhase LlvmLlc input_fn dflags    = do      let lc_opts = getOpts dflags opt_lc -    let opt_lvl = max 0 (min 2 $ optLevel dflags) -#if darwin_TARGET_OS -    let nphase = LlvmMangle -#else -    let nphase = As -#endif -    let rmodel | opt_PIC        = "pic" +        opt_lvl = max 0 (min 2 $ optLevel dflags) +        rmodel | opt_PIC        = "pic"                 | not opt_Static = "dynamic-no-pic"                 | otherwise      = "static" -    output_fn <- phaseOutputFilename nphase +    output_fn <- phaseOutputFilename LlvmMangle      io $ SysTools.runLlvmLlc dflags                  ([ SysTools.Option (llvmOpts !! opt_lvl), @@ -1334,14 +1327,12 @@ runPhase LlvmLlc input_fn dflags                      SysTools.Option "-o", SysTools.FileOption "" output_fn]                  ++ map SysTools.Option lc_opts) -    return (nphase, output_fn) +    return (LlvmMangle, output_fn)    where -#if darwin_TARGET_OS -        llvmOpts = ["-O1", "-O2", "-O2"] -#else -        llvmOpts = ["-O1", "-O2", "-O3"] -#endif - +        -- Bug in LLVM at O3 on OSX. +        llvmOpts = if cTargetOS == OSX +                   then ["-O1", "-O2", "-O2"] +                   else ["-O1", "-O2", "-O3"]  -----------------------------------------------------------------------------  -- LlvmMangle phase @@ -1654,11 +1645,12 @@ linkBinary dflags o_files dep_packages = do                        ++ map SysTools.Option (                           [] -#ifdef mingw32_TARGET_OS                        -- Permit the linker to auto link _symbol to _imp_symbol.                        -- This lets us link against DLLs without needing an "import library". -                      ++ ["-Wl,--enable-auto-import"] -#endif +                      ++ (if cTargetOS == Windows +                          then ["-Wl,--enable-auto-import"] +                          else []) +                        ++ o_files                        ++ extra_ld_inputs                        ++ lib_path_opts @@ -1689,19 +1681,15 @@ linkBinary dflags o_files dep_packages = do  exeFileName :: DynFlags -> FilePath  exeFileName dflags    | Just s <- outputFile dflags = -#if defined(mingw32_HOST_OS) -      if null (takeExtension s) -        then s <.> "exe" -        else s -#else -      s -#endif +      if cTargetOS == Windows +      then if null (takeExtension s) +           then s <.> "exe" +           else s +      else s    | otherwise = -#if defined(mingw32_HOST_OS) -        "main.exe" -#else -        "a.out" -#endif +      if cTargetOS == Windows +      then "main.exe" +      else "a.out"  maybeCreateManifest     :: DynFlags diff --git a/compiler/main/DynFlags.hs b/compiler/main/DynFlags.hs index df75762e21..cdbaa53adb 100644 --- a/compiler/main/DynFlags.hs +++ b/compiler/main/DynFlags.hs @@ -77,9 +77,7 @@ module DynFlags (  #include "HsVersions.h" -#ifndef OMIT_NATIVE_CODEGEN  import Platform -#endif  import Module  import PackageConfig  import PrelNames        ( mAIN ) @@ -110,7 +108,7 @@ import Data.Char  import Data.List  import Data.Map (Map)  import qualified Data.Map as Map --- import Data.Maybe +import Distribution.System  import System.FilePath  import System.IO        ( stderr, hPutChar ) @@ -360,6 +358,7 @@ data ExtensionFlag     | Opt_KindSignatures     | Opt_ParallelListComp     | Opt_TransformListComp +   | Opt_MonadComprehensions     | Opt_GeneralizedNewtypeDeriving     | Opt_RecursiveDo     | Opt_DoRec @@ -402,9 +401,7 @@ data DynFlags = DynFlags {    floatLamArgs          :: Maybe Int,   -- ^ Arg count for lambda floating    			   	 	--   See CoreMonad.FloatOutSwitches -#ifndef OMIT_NATIVE_CODEGEN -  targetPlatform	:: Platform,	-- ^ The platform we're compiling for. Used by the NCG. -#endif +  targetPlatform        :: Platform.Platform, -- ^ The platform we're compiling for. Used by the NCG.    cmdlineHcIncludes     :: [String],    -- ^ @\-\#includes@    importPaths           :: [FilePath],    mainModIs             :: Module, @@ -630,6 +627,14 @@ data HscTarget    | HscNothing     -- ^ Don't generate any code.  See notes above.    deriving (Eq, Show) +showHscTargetFlag :: HscTarget -> String +showHscTargetFlag HscC           = "-fvia-c" +showHscTargetFlag HscAsm         = "-fasm" +showHscTargetFlag HscLlvm        = "-fllvm" +showHscTargetFlag HscJava        = panic "No flag for HscJava" +showHscTargetFlag HscInterpreted = "-fbyte-code" +showHscTargetFlag HscNothing     = "-fno-code" +  -- | Will this target result in an object file on the disk?  isObjectTarget :: HscTarget -> Bool  isObjectTarget HscC     = True @@ -692,8 +697,9 @@ defaultHscTarget = defaultObjectTarget  -- object files on the current platform.  defaultObjectTarget :: HscTarget  defaultObjectTarget +  | cGhcUnregisterised    == "YES"      =  HscC    | cGhcWithNativeCodeGen == "YES"      =  HscAsm -  | otherwise                           =  HscC +  | otherwise                           =  HscLlvm  data DynLibLoader    = Deployable @@ -740,9 +746,7 @@ defaultDynFlags mySettings =          floatLamArgs            = Just 0,	-- Default: float only if no fvs          strictnessBefore        = [], -#ifndef OMIT_NATIVE_CODEGEN          targetPlatform          = defaultTargetPlatform, -#endif          cmdlineHcIncludes       = [],          importPaths             = ["."],          mainModIs               = mAIN, @@ -1099,12 +1103,14 @@ parseDynamicFlags_ dflags0 args pkg_flags = do    when (not (null errs)) $ ghcError $ errorsToGhcException errs    let (pic_warns, dflags2) -#if !(x86_64_TARGET_ARCH && linux_TARGET_OS) -        | (not opt_Static || opt_PIC) && hscTarget dflags1 == HscLlvm -        = ([L noSrcSpan $ "Warning: -fllvm is incompatible with -fPIC and -" -                ++ "dynamic on this platform;\n              ignoring -fllvm"], -                dflags1{ hscTarget = HscAsm }) -#endif +        | not (cTargetArch == X86_64 && (cTargetOS == Linux || cTargetOS == OSX)) && +          (not opt_Static || opt_PIC) && +          hscTarget dflags1 == HscLlvm +        = ([L noSrcSpan $ "Warning: -fllvm is incompatible with -fPIC and " +                       ++ "-dynamic on this platform;\n" +                       ++ "         using " +                       ++ showHscTargetFlag defaultObjectTarget ++ " instead"], +                dflags1{ hscTarget = defaultObjectTarget })          | otherwise = ([], dflags1)    return (dflags2, leftover, pic_warns ++ warns) @@ -1345,10 +1351,11 @@ dynamic_flags = [    , Flag "w"      (NoArg (mapM_ unSetDynFlag minuswRemovesOpts))          ------ Optimisation flags ------------------------------------------ -  , Flag "O"      (noArg (setOptLevel 1)) -  , Flag "Onot"   (noArgDF (setOptLevel 0) "Use -O0 instead") -  , Flag "Odph"   (noArg setDPHOpt) -  , Flag "O"      (OptIntSuffix (\mb_n -> upd (setOptLevel (mb_n `orElse` 1)))) +  , Flag "O"      (noArgM (setOptLevel 1)) +  , Flag "Onot"   (noArgM (\dflags -> do deprecate "Use -O0 instead" +                                         setOptLevel 0 dflags)) +  , Flag "Odph"   (noArgM setDPHOpt) +  , Flag "O"      (optIntSuffixM (\mb_n -> setOptLevel (mb_n `orElse` 1)))                  -- If the number is missing, use 1    , Flag "fsimplifier-phases"          (intSuffix (\n d -> d{ simplPhases = n })) @@ -1615,6 +1622,7 @@ xFlags = [    ( "EmptyDataDecls",                   Opt_EmptyDataDecls, nop ),    ( "ParallelListComp",                 Opt_ParallelListComp, nop ),    ( "TransformListComp",                Opt_TransformListComp, nop ), +  ( "MonadComprehensions",              Opt_MonadComprehensions, nop),    ( "ForeignFunctionInterface",         Opt_ForeignFunctionInterface, nop ),    ( "UnliftedFFITypes",                 Opt_UnliftedFFITypes, nop ),    ( "GHCForeignImportPrim",             Opt_GHCForeignImportPrim, nop ), @@ -1623,9 +1631,9 @@ xFlags = [    ( "RankNTypes",                       Opt_RankNTypes, nop ),    ( "ImpredicativeTypes",               Opt_ImpredicativeTypes, nop),     ( "TypeOperators",                    Opt_TypeOperators, nop ), -  ( "RecursiveDo",                      Opt_RecursiveDo, +  ( "RecursiveDo",                      Opt_RecursiveDo,     -- Enables 'mdo'      deprecatedForExtension "DoRec"), -  ( "DoRec",                            Opt_DoRec, nop ), +  ( "DoRec",                            Opt_DoRec, nop ),    -- Enables 'rec' keyword     ( "Arrows",                           Opt_Arrows, nop ),    ( "ParallelArrays",                   Opt_ParallelArrays, nop ),    ( "TemplateHaskell",                  Opt_TemplateHaskell, checkTemplateHaskellOk ), @@ -1904,13 +1912,21 @@ checkTemplateHaskellOk _ = return ()  type DynP = EwM (CmdLineP DynFlags)  upd :: (DynFlags -> DynFlags) -> DynP () -upd f = liftEwM (do { dfs <- getCmdLineState -                    ; putCmdLineState $! (f dfs) }) +upd f = liftEwM (do dflags <- getCmdLineState +                    putCmdLineState $! f dflags) + +updM :: (DynFlags -> DynP DynFlags) -> DynP () +updM f = do dflags <- liftEwM getCmdLineState +            dflags' <- f dflags +            liftEwM $ putCmdLineState $! dflags'  --------------- Constructor functions for OptKind -----------------  noArg :: (DynFlags -> DynFlags) -> OptKind (CmdLineP DynFlags)  noArg fn = NoArg (upd fn) +noArgM :: (DynFlags -> DynP DynFlags) -> OptKind (CmdLineP DynFlags) +noArgM fn = NoArg (updM fn) +  noArgDF :: (DynFlags -> DynFlags) -> String -> OptKind (CmdLineP DynFlags)  noArgDF fn deprec = NoArg (upd fn >> deprecate deprec) @@ -1924,6 +1940,10 @@ hasArgDF fn deprec = HasArg (\s -> do { upd (fn s)  intSuffix :: (Int -> DynFlags -> DynFlags) -> OptKind (CmdLineP DynFlags)  intSuffix fn = IntSuffix (\n -> upd (fn n)) +optIntSuffixM :: (Maybe Int -> DynFlags -> DynP DynFlags) +              -> OptKind (CmdLineP DynFlags) +optIntSuffixM fn = OptIntSuffix (\mi -> updM (fn mi)) +  setDumpFlag :: DynFlag -> OptKind (CmdLineP DynFlags)  setDumpFlag dump_flag = NoArg (setDumpFlag' dump_flag) @@ -2021,20 +2041,36 @@ setTarget l = upd set  -- not from bytecode to object-code.  The idea is that -fasm/-fllvm  -- can be safely used in an OPTIONS_GHC pragma.  setObjTarget :: HscTarget -> DynP () -setObjTarget l = upd set +setObjTarget l = updM set    where -   set dfs -     | isObjectTarget (hscTarget dfs) = dfs { hscTarget = l } -     | otherwise = dfs - -setOptLevel :: Int -> DynFlags -> DynFlags +   set dflags +     | isObjectTarget (hscTarget dflags) +       = case l of +         HscC +          | cGhcUnregisterised /= "YES" -> +             do addWarn ("Compiler not unregisterised, so ignoring " ++ +                         showHscTargetFlag l) +                return dflags +         HscAsm +          | cGhcWithNativeCodeGen /= "YES" -> +             do addWarn ("Compiler has no native codegen, so ignoring " ++ +                         showHscTargetFlag l) +                return dflags +         HscLlvm +          | cGhcUnregisterised == "YES" -> +             do addWarn ("Compiler unregisterised, so ignoring " ++ +                         showHscTargetFlag l) +                return dflags +         _ -> return $ dflags { hscTarget = l } +     | otherwise = return dflags + +setOptLevel :: Int -> DynFlags -> DynP DynFlags  setOptLevel n dflags     | hscTarget dflags == HscInterpreted && n > 0 -        = dflags -            -- not in IO any more, oh well: -            -- putStr "warning: -O conflicts with --interactive; -O ignored.\n" +        = do addWarn "-O conflicts with --interactive; -O ignored." +             return dflags     | otherwise -        = updOptLevel n dflags +        = return (updOptLevel n dflags)  -- -Odph is equivalent to @@ -2043,7 +2079,7 @@ setOptLevel n dflags  --    -fmax-simplifier-iterations20     this is necessary sometimes  --    -fsimplifier-phases=3             we use an additional simplifier phase for fusion  -- -setDPHOpt :: DynFlags -> DynFlags +setDPHOpt :: DynFlags -> DynP DynFlags  setDPHOpt dflags = setOptLevel 2 (dflags { maxSimplIterations  = 20                                           , simplPhases         = 3                                           }) diff --git a/compiler/main/GhcMonad.hs b/compiler/main/GhcMonad.hs index 711259c9ba..4c72f144c2 100644 --- a/compiler/main/GhcMonad.hs +++ b/compiler/main/GhcMonad.hs @@ -15,11 +15,11 @@ module GhcMonad (          reflectGhc, reifyGhc,          getSessionDynFlags,           liftIO, -	Session(..), withSession, modifySession, withTempSession, +        Session(..), withSession, modifySession, withTempSession,          -- ** Warnings          logWarnings, printException, printExceptionAndWarnings, -	WarnErrLogger, defaultWarnErrLogger +        WarnErrLogger, defaultWarnErrLogger    ) where  import MonadUtils diff --git a/compiler/main/HscMain.lhs b/compiler/main/HscMain.lhs index 36e53a83f9..6a5552f5df 100644 --- a/compiler/main/HscMain.lhs +++ b/compiler/main/HscMain.lhs @@ -1132,7 +1132,7 @@ hscTcExpr	-- Typecheck an expression (but don't run it)  hscTcExpr hsc_env expr = runHsc hsc_env $ do      maybe_stmt <- hscParseStmt expr      case maybe_stmt of -        Just (L _ (ExprStmt expr _ _)) -> +        Just (L _ (ExprStmt expr _ _ _)) ->              ioMsgMaybe $ tcRnExpr hsc_env (hsc_IC hsc_env) expr          _ ->              liftIO $ throwIO $ mkSrcErr $ unitBag $ mkPlainErrMsg noSrcSpan diff --git a/compiler/main/SysTools.lhs b/compiler/main/SysTools.lhs index 97a6514746..436cfa6c4c 100644 --- a/compiler/main/SysTools.lhs +++ b/compiler/main/SysTools.lhs @@ -238,7 +238,7 @@ initSysTools mbMinusB                  ld_prog  = gcc_prog                  ld_args  = gcc_args -        -- figure out llvm location. (TODO: Acutally implement). +        -- We just assume on command line          ; let lc_prog = "llc"                lo_prog = "opt" diff --git a/compiler/nativeGen/Alpha/CodeGen.hs b/compiler/nativeGen/Alpha/CodeGen.hs deleted file mode 100644 index 4ce774f14f..0000000000 --- a/compiler/nativeGen/Alpha/CodeGen.hs +++ /dev/null @@ -1,789 +0,0 @@ -module Alpha.CodeGen () - -where - -{- - -getRegister :: CmmExpr -> NatM Register - -#if !x86_64_TARGET_ARCH -    -- on x86_64, we have %rip for PicBaseReg, but it's not a full-featured -    -- register, it can only be used for rip-relative addressing. -getRegister (CmmReg (CmmGlobal PicBaseReg)) -  = do -      reg <- getPicBaseNat wordSize -      return (Fixed wordSize reg nilOL) -#endif - -getRegister (CmmReg reg)  -  = return (Fixed (cmmTypeSize (cmmRegType reg))  -		  (getRegisterReg reg) nilOL) - -getRegister tree@(CmmRegOff _ _)  -  = getRegister (mangleIndexTree tree) - - -#if WORD_SIZE_IN_BITS==32 -    -- for 32-bit architectuers, support some 64 -> 32 bit conversions: -    -- TO_W_(x), TO_W_(x >> 32) - -getRegister (CmmMachOp (MO_UU_Conv W64 W32) -             [CmmMachOp (MO_U_Shr W64) [x,CmmLit (CmmInt 32 _)]]) = do -  ChildCode64 code rlo <- iselExpr64 x -  return $ Fixed II32 (getHiVRegFromLo rlo) code - -getRegister (CmmMachOp (MO_SS_Conv W64 W32) -             [CmmMachOp (MO_U_Shr W64) [x,CmmLit (CmmInt 32 _)]]) = do -  ChildCode64 code rlo <- iselExpr64 x -  return $ Fixed II32 (getHiVRegFromLo rlo) code - -getRegister (CmmMachOp (MO_UU_Conv W64 W32) [x]) = do -  ChildCode64 code rlo <- iselExpr64 x -  return $ Fixed II32 rlo code - -getRegister (CmmMachOp (MO_SS_Conv W64 W32) [x]) = do -  ChildCode64 code rlo <- iselExpr64 x -  return $ Fixed II32 rlo code        - -#endif - --- end of machine-"independent" bit; here we go on the rest... - - -getRegister (StDouble d) -  = getBlockIdNat 	    	    `thenNat` \ lbl -> -    getNewRegNat PtrRep    	    `thenNat` \ tmp -> -    let code dst = mkSeqInstrs [ -	    LDATA RoDataSegment lbl [ -		    DATA TF [ImmLab (rational d)] -		], -	    LDA tmp (AddrImm (ImmCLbl lbl)), -	    LD TF dst (AddrReg tmp)] -    in -    	return (Any FF64 code) - -getRegister (StPrim primop [x]) -- unary PrimOps -  = case primop of -      IntNegOp -> trivialUCode (NEG Q False) x - -      NotOp    -> trivialUCode NOT x - -      FloatNegOp  -> trivialUFCode FloatRep (FNEG TF) x -      DoubleNegOp -> trivialUFCode FF64 (FNEG TF) x - -      OrdOp -> coerceIntCode IntRep x -      ChrOp -> chrCode x - -      Float2IntOp  -> coerceFP2Int    x -      Int2FloatOp  -> coerceInt2FP pr x -      Double2IntOp -> coerceFP2Int    x -      Int2DoubleOp -> coerceInt2FP pr x - -      Double2FloatOp -> coerceFltCode x -      Float2DoubleOp -> coerceFltCode x - -      other_op -> getRegister (StCall fn CCallConv FF64 [x]) -	where -	  fn = case other_op of -		 FloatExpOp    -> fsLit "exp" -		 FloatLogOp    -> fsLit "log" -		 FloatSqrtOp   -> fsLit "sqrt" -		 FloatSinOp    -> fsLit "sin" -		 FloatCosOp    -> fsLit "cos" -		 FloatTanOp    -> fsLit "tan" -		 FloatAsinOp   -> fsLit "asin" -		 FloatAcosOp   -> fsLit "acos" -		 FloatAtanOp   -> fsLit "atan" -		 FloatSinhOp   -> fsLit "sinh" -		 FloatCoshOp   -> fsLit "cosh" -		 FloatTanhOp   -> fsLit "tanh" -		 DoubleExpOp   -> fsLit "exp" -		 DoubleLogOp   -> fsLit "log" -		 DoubleSqrtOp  -> fsLit "sqrt" -		 DoubleSinOp   -> fsLit "sin" -		 DoubleCosOp   -> fsLit "cos" -		 DoubleTanOp   -> fsLit "tan" -		 DoubleAsinOp  -> fsLit "asin" -		 DoubleAcosOp  -> fsLit "acos" -		 DoubleAtanOp  -> fsLit "atan" -		 DoubleSinhOp  -> fsLit "sinh" -		 DoubleCoshOp  -> fsLit "cosh" -		 DoubleTanhOp  -> fsLit "tanh" -  where -    pr = panic "MachCode.getRegister: no primrep needed for Alpha" - -getRegister (StPrim primop [x, y]) -- dyadic PrimOps -  = case primop of -      CharGtOp -> trivialCode (CMP LTT) y x -      CharGeOp -> trivialCode (CMP LE) y x -      CharEqOp -> trivialCode (CMP EQQ) x y -      CharNeOp -> int_NE_code x y -      CharLtOp -> trivialCode (CMP LTT) x y -      CharLeOp -> trivialCode (CMP LE) x y - -      IntGtOp  -> trivialCode (CMP LTT) y x -      IntGeOp  -> trivialCode (CMP LE) y x -      IntEqOp  -> trivialCode (CMP EQQ) x y -      IntNeOp  -> int_NE_code x y -      IntLtOp  -> trivialCode (CMP LTT) x y -      IntLeOp  -> trivialCode (CMP LE) x y - -      WordGtOp -> trivialCode (CMP ULT) y x -      WordGeOp -> trivialCode (CMP ULE) x y -      WordEqOp -> trivialCode (CMP EQQ)  x y -      WordNeOp -> int_NE_code x y -      WordLtOp -> trivialCode (CMP ULT) x y -      WordLeOp -> trivialCode (CMP ULE) x y - -      AddrGtOp -> trivialCode (CMP ULT) y x -      AddrGeOp -> trivialCode (CMP ULE) y x -      AddrEqOp -> trivialCode (CMP EQQ)  x y -      AddrNeOp -> int_NE_code x y -      AddrLtOp -> trivialCode (CMP ULT) x y -      AddrLeOp -> trivialCode (CMP ULE) x y -	 -      FloatGtOp -> cmpF_code (FCMP TF LE) EQQ x y -      FloatGeOp -> cmpF_code (FCMP TF LTT) EQQ x y -      FloatEqOp -> cmpF_code (FCMP TF EQQ) NE x y -      FloatNeOp -> cmpF_code (FCMP TF EQQ) EQQ x y -      FloatLtOp -> cmpF_code (FCMP TF LTT) NE x y -      FloatLeOp -> cmpF_code (FCMP TF LE) NE x y - -      DoubleGtOp -> cmpF_code (FCMP TF LE) EQQ x y -      DoubleGeOp -> cmpF_code (FCMP TF LTT) EQQ x y -      DoubleEqOp -> cmpF_code (FCMP TF EQQ) NE x y -      DoubleNeOp -> cmpF_code (FCMP TF EQQ) EQQ x y -      DoubleLtOp -> cmpF_code (FCMP TF LTT) NE x y -      DoubleLeOp -> cmpF_code (FCMP TF LE) NE x y - -      IntAddOp  -> trivialCode (ADD Q False) x y -      IntSubOp  -> trivialCode (SUB Q False) x y -      IntMulOp  -> trivialCode (MUL Q False) x y -      IntQuotOp -> trivialCode (DIV Q False) x y -      IntRemOp  -> trivialCode (REM Q False) x y - -      WordAddOp  -> trivialCode (ADD Q False) x y -      WordSubOp  -> trivialCode (SUB Q False) x y -      WordMulOp  -> trivialCode (MUL Q False) x y -      WordQuotOp -> trivialCode (DIV Q True) x y -      WordRemOp  -> trivialCode (REM Q True) x y - -      FloatAddOp -> trivialFCode  W32 (FADD TF) x y -      FloatSubOp -> trivialFCode  W32 (FSUB TF) x y -      FloatMulOp -> trivialFCode  W32 (FMUL TF) x y -      FloatDivOp -> trivialFCode  W32 (FDIV TF) x y - -      DoubleAddOp -> trivialFCode  W64 (FADD TF) x y -      DoubleSubOp -> trivialFCode  W64 (FSUB TF) x y -      DoubleMulOp -> trivialFCode  W64 (FMUL TF) x y -      DoubleDivOp -> trivialFCode  W64 (FDIV TF) x y - -      AddrAddOp  -> trivialCode (ADD Q False) x y -      AddrSubOp  -> trivialCode (SUB Q False) x y -      AddrRemOp  -> trivialCode (REM Q True) x y - -      AndOp  -> trivialCode AND x y -      OrOp   -> trivialCode OR  x y -      XorOp  -> trivialCode XOR x y -      SllOp  -> trivialCode SLL x y -      SrlOp  -> trivialCode SRL x y - -      ISllOp -> trivialCode SLL x y -- was: panic "AlphaGen:isll" -      ISraOp -> trivialCode SRA x y -- was: panic "AlphaGen:isra" -      ISrlOp -> trivialCode SRL x y -- was: panic "AlphaGen:isrl" - -      FloatPowerOp  -> getRegister (StCall (fsLit "pow") CCallConv FF64 [x,y]) -      DoublePowerOp -> getRegister (StCall (fsLit "pow") CCallConv FF64 [x,y]) -  where -    {- ------------------------------------------------------------ -	Some bizarre special code for getting condition codes into -	registers.  Integer non-equality is a test for equality -	followed by an XOR with 1.  (Integer comparisons always set -	the result register to 0 or 1.)  Floating point comparisons of -	any kind leave the result in a floating point register, so we -	need to wrangle an integer register out of things. -    -} -    int_NE_code :: StixTree -> StixTree -> NatM Register - -    int_NE_code x y -      = trivialCode (CMP EQQ) x y	`thenNat` \ register -> -	getNewRegNat IntRep		`thenNat` \ tmp -> -	let -	    code = registerCode register tmp -	    src  = registerName register tmp -	    code__2 dst = code . mkSeqInstr (XOR src (RIImm (ImmInt 1)) dst) -	in -	return (Any IntRep code__2) - -    {- ------------------------------------------------------------ -	Comments for int_NE_code also apply to cmpF_code -    -} -    cmpF_code -	:: (Reg -> Reg -> Reg -> Instr) -	-> Cond -	-> StixTree -> StixTree -	-> NatM Register - -    cmpF_code instr cond x y -      = trivialFCode pr instr x y	`thenNat` \ register -> -	getNewRegNat FF64		`thenNat` \ tmp -> -	getBlockIdNat			`thenNat` \ lbl -> -	let -	    code = registerCode register tmp -	    result  = registerName register tmp - -	    code__2 dst = code . mkSeqInstrs [ -		OR zeroh (RIImm (ImmInt 1)) dst, -		BF cond  result (ImmCLbl lbl), -		OR zeroh (RIReg zeroh) dst, -		NEWBLOCK lbl] -	in -	return (Any IntRep code__2) -      where -	pr = panic "trivialU?FCode: does not use PrimRep on Alpha" -      ------------------------------------------------------------ - -getRegister (CmmLoad pk mem) -  = getAmode mem    	    	    `thenNat` \ amode -> -    let -    	code = amodeCode amode -    	src   = amodeAddr amode -    	size = primRepToSize pk -    	code__2 dst = code . mkSeqInstr (LD size dst src) -    in -    return (Any pk code__2) - -getRegister (StInt i) -  | fits8Bits i -  = let -    	code dst = mkSeqInstr (OR zeroh (RIImm src) dst) -    in -    return (Any IntRep code) -  | otherwise -  = let -    	code dst = mkSeqInstr (LDI Q dst src) -    in -    return (Any IntRep code) -  where -    src = ImmInt (fromInteger i) - -getRegister leaf -  | isJust imm -  = let -    	code dst = mkSeqInstr (LDA dst (AddrImm imm__2)) -    in -    return (Any PtrRep code) -  where -    imm = maybeImm leaf -    imm__2 = case imm of Just x -> x - - -getAmode :: CmmExpr -> NatM Amode -getAmode tree@(CmmRegOff _ _) = getAmode (mangleIndexTree tree) - --- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -#if alpha_TARGET_ARCH - -getAmode (StPrim IntSubOp [x, StInt i]) -  = getNewRegNat PtrRep		`thenNat` \ tmp -> -    getRegister x		`thenNat` \ register -> -    let -    	code = registerCode register tmp -    	reg  = registerName register tmp -    	off  = ImmInt (-(fromInteger i)) -    in -    return (Amode (AddrRegImm reg off) code) - -getAmode (StPrim IntAddOp [x, StInt i]) -  = getNewRegNat PtrRep		`thenNat` \ tmp -> -    getRegister x		`thenNat` \ register -> -    let -    	code = registerCode register tmp -    	reg  = registerName register tmp -    	off  = ImmInt (fromInteger i) -    in -    return (Amode (AddrRegImm reg off) code) - -getAmode leaf -  | isJust imm -  = return (Amode (AddrImm imm__2) id) -  where -    imm = maybeImm leaf -    imm__2 = case imm of Just x -> x - -getAmode other -  = getNewRegNat PtrRep		`thenNat` \ tmp -> -    getRegister other		`thenNat` \ register -> -    let -    	code = registerCode register tmp -    	reg  = registerName register tmp -    in -    return (Amode (AddrReg reg) code) - -#endif /* alpha_TARGET_ARCH */ - - --- ----------------------------------------------------------------------------- --- Generating assignments - --- Assignments are really at the heart of the whole code generation --- business.  Almost all top-level nodes of any real importance are --- assignments, which correspond to loads, stores, or register --- transfers.  If we're really lucky, some of the register transfers --- will go away, because we can use the destination register to --- complete the code generation for the right hand side.  This only --- fails when the right hand side is forced into a fixed register --- (e.g. the result of a call). - -assignMem_IntCode :: Size -> CmmExpr -> CmmExpr -> NatM InstrBlock -assignReg_IntCode :: Size -> CmmReg  -> CmmExpr -> NatM InstrBlock - -assignMem_FltCode :: Size -> CmmExpr -> CmmExpr -> NatM InstrBlock -assignReg_FltCode :: Size -> CmmReg  -> CmmExpr -> NatM InstrBlock - - -assignIntCode pk (CmmLoad dst _) src -  = getNewRegNat IntRep    	    `thenNat` \ tmp -> -    getAmode dst    	    	    `thenNat` \ amode -> -    getRegister src	     	    `thenNat` \ register -> -    let -    	code1   = amodeCode amode [] -    	dst__2  = amodeAddr amode -    	code2   = registerCode register tmp [] -    	src__2  = registerName register tmp -    	sz      = primRepToSize pk -    	code__2 = asmSeqThen [code1, code2] . mkSeqInstr (ST sz src__2 dst__2) -    in -    return code__2 - -assignIntCode pk dst src -  = getRegister dst	    	    	    `thenNat` \ register1 -> -    getRegister src	    	    	    `thenNat` \ register2 -> -    let -    	dst__2  = registerName register1 zeroh -    	code    = registerCode register2 dst__2 -    	src__2  = registerName register2 dst__2 -    	code__2 = if isFixed register2 -		  then code . mkSeqInstr (OR src__2 (RIReg src__2) dst__2) -    	    	  else code -    in -    return code__2 - -assignFltCode pk (CmmLoad dst _) src -  = getNewRegNat pk        	    `thenNat` \ tmp -> -    getAmode dst    	    	    `thenNat` \ amode -> -    getRegister src	    	    	    `thenNat` \ register -> -    let -    	code1   = amodeCode amode [] -    	dst__2  = amodeAddr amode -    	code2   = registerCode register tmp [] -    	src__2  = registerName register tmp -    	sz      = primRepToSize pk -    	code__2 = asmSeqThen [code1, code2] . mkSeqInstr (ST sz src__2 dst__2) -    in -    return code__2 - -assignFltCode pk dst src -  = getRegister dst	    	    	    `thenNat` \ register1 -> -    getRegister src	    	    	    `thenNat` \ register2 -> -    let -    	dst__2  = registerName register1 zeroh -    	code    = registerCode register2 dst__2 -    	src__2  = registerName register2 dst__2 -    	code__2 = if isFixed register2 -		  then code . mkSeqInstr (FMOV src__2 dst__2) -		  else code -    in -    return code__2 - - --- ----------------------------------------------------------------------------- --- Generating an non-local jump - --- (If applicable) Do not fill the delay slots here; you will confuse the --- register allocator. - -genJump :: CmmExpr{-the branch target-} -> NatM InstrBlock - --- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -genJump (CmmLabel lbl) -  | isAsmTemp lbl = returnInstr (BR target) -  | otherwise     = returnInstrs [LDA pv (AddrImm target), JMP zeroh (AddrReg pv) 0] -  where -    target = ImmCLbl lbl - -genJump tree -  = getRegister tree	     	    `thenNat` \ register -> -    getNewRegNat PtrRep    	    `thenNat` \ tmp -> -    let -    	dst    = registerName register pv -    	code   = registerCode register pv -    	target = registerName register pv -    in -    if isFixed register then -	returnSeq code [OR dst (RIReg dst) pv, JMP zeroh (AddrReg pv) 0] -    else -    return (code . mkSeqInstr (JMP zeroh (AddrReg pv) 0)) - - --- ----------------------------------------------------------------------------- ---  Unconditional branches - -genBranch :: BlockId -> NatM InstrBlock - -genBranch = return . toOL . mkBranchInstr - - --- ----------------------------------------------------------------------------- ---  Conditional jumps - -{- -Conditional jumps are always to local labels, so we can use branch -instructions.  We peek at the arguments to decide what kind of -comparison to do. - -ALPHA: For comparisons with 0, we're laughing, because we can just do -the desired conditional branch. - --} - - -genCondJump -    :: BlockId	    -- the branch target -    -> CmmExpr      -- the condition on which to branch -    -> NatM InstrBlock - --- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -genCondJump id (StPrim op [x, StInt 0]) -  = getRegister x	  	    	    `thenNat` \ register -> -    getNewRegNat (registerRep register) -    	    	        	    `thenNat` \ tmp -> -    let -    	code   = registerCode register tmp -    	value  = registerName register tmp -    	pk     = registerRep register -	target = ImmCLbl lbl -    in -    returnSeq code [BI (cmpOp op) value target] -  where -    cmpOp CharGtOp = GTT -    cmpOp CharGeOp = GE -    cmpOp CharEqOp = EQQ -    cmpOp CharNeOp = NE -    cmpOp CharLtOp = LTT -    cmpOp CharLeOp = LE -    cmpOp IntGtOp = GTT -    cmpOp IntGeOp = GE -    cmpOp IntEqOp = EQQ -    cmpOp IntNeOp = NE -    cmpOp IntLtOp = LTT -    cmpOp IntLeOp = LE -    cmpOp WordGtOp = NE -    cmpOp WordGeOp = ALWAYS -    cmpOp WordEqOp = EQQ -    cmpOp WordNeOp = NE -    cmpOp WordLtOp = NEVER -    cmpOp WordLeOp = EQQ -    cmpOp AddrGtOp = NE -    cmpOp AddrGeOp = ALWAYS -    cmpOp AddrEqOp = EQQ -    cmpOp AddrNeOp = NE -    cmpOp AddrLtOp = NEVER -    cmpOp AddrLeOp = EQQ - -genCondJump lbl (StPrim op [x, StDouble 0.0]) -  = getRegister x	  	    	    `thenNat` \ register -> -    getNewRegNat (registerRep register) -    	    	        	    `thenNat` \ tmp -> -    let -    	code   = registerCode register tmp -    	value  = registerName register tmp -    	pk     = registerRep register -	target = ImmCLbl lbl -    in -    return (code . mkSeqInstr (BF (cmpOp op) value target)) -  where -    cmpOp FloatGtOp = GTT -    cmpOp FloatGeOp = GE -    cmpOp FloatEqOp = EQQ -    cmpOp FloatNeOp = NE -    cmpOp FloatLtOp = LTT -    cmpOp FloatLeOp = LE -    cmpOp DoubleGtOp = GTT -    cmpOp DoubleGeOp = GE -    cmpOp DoubleEqOp = EQQ -    cmpOp DoubleNeOp = NE -    cmpOp DoubleLtOp = LTT -    cmpOp DoubleLeOp = LE - -genCondJump lbl (StPrim op [x, y]) -  | fltCmpOp op -  = trivialFCode pr instr x y 	    `thenNat` \ register -> -    getNewRegNat FF64    	    `thenNat` \ tmp -> -    let -    	code   = registerCode register tmp -    	result = registerName register tmp -	target = ImmCLbl lbl -    in -    return (code . mkSeqInstr (BF cond result target)) -  where -    pr = panic "trivialU?FCode: does not use PrimRep on Alpha" - -    fltCmpOp op = case op of -	FloatGtOp -> True -	FloatGeOp -> True -	FloatEqOp -> True -	FloatNeOp -> True -	FloatLtOp -> True -	FloatLeOp -> True -	DoubleGtOp -> True -	DoubleGeOp -> True -	DoubleEqOp -> True -	DoubleNeOp -> True -	DoubleLtOp -> True -	DoubleLeOp -> True -	_ -> False -    (instr, cond) = case op of -	FloatGtOp -> (FCMP TF LE, EQQ) -	FloatGeOp -> (FCMP TF LTT, EQQ) -	FloatEqOp -> (FCMP TF EQQ, NE) -	FloatNeOp -> (FCMP TF EQQ, EQQ) -	FloatLtOp -> (FCMP TF LTT, NE) -	FloatLeOp -> (FCMP TF LE, NE) -	DoubleGtOp -> (FCMP TF LE, EQQ) -	DoubleGeOp -> (FCMP TF LTT, EQQ) -	DoubleEqOp -> (FCMP TF EQQ, NE) -	DoubleNeOp -> (FCMP TF EQQ, EQQ) -	DoubleLtOp -> (FCMP TF LTT, NE) -	DoubleLeOp -> (FCMP TF LE, NE) - -genCondJump lbl (StPrim op [x, y]) -  = trivialCode instr x y    	    `thenNat` \ register -> -    getNewRegNat IntRep    	    `thenNat` \ tmp -> -    let -    	code   = registerCode register tmp -    	result = registerName register tmp -	target = ImmCLbl lbl -    in -    return (code . mkSeqInstr (BI cond result target)) -  where -    (instr, cond) = case op of -	CharGtOp -> (CMP LE, EQQ) -	CharGeOp -> (CMP LTT, EQQ) -	CharEqOp -> (CMP EQQ, NE) -	CharNeOp -> (CMP EQQ, EQQ) -	CharLtOp -> (CMP LTT, NE) -	CharLeOp -> (CMP LE, NE) -	IntGtOp -> (CMP LE, EQQ) -	IntGeOp -> (CMP LTT, EQQ) -	IntEqOp -> (CMP EQQ, NE) -	IntNeOp -> (CMP EQQ, EQQ) -	IntLtOp -> (CMP LTT, NE) -	IntLeOp -> (CMP LE, NE) -	WordGtOp -> (CMP ULE, EQQ) -	WordGeOp -> (CMP ULT, EQQ) -	WordEqOp -> (CMP EQQ, NE) -	WordNeOp -> (CMP EQQ, EQQ) -	WordLtOp -> (CMP ULT, NE) -	WordLeOp -> (CMP ULE, NE) -	AddrGtOp -> (CMP ULE, EQQ) -	AddrGeOp -> (CMP ULT, EQQ) -	AddrEqOp -> (CMP EQQ, NE) -	AddrNeOp -> (CMP EQQ, EQQ) -	AddrLtOp -> (CMP ULT, NE) -	AddrLeOp -> (CMP ULE, NE) - --- ----------------------------------------------------------------------------- ---  Generating C calls - --- Now the biggest nightmare---calls.  Most of the nastiness is buried in --- @get_arg@, which moves the arguments to the correct registers/stack --- locations.  Apart from that, the code is easy. ---  --- (If applicable) Do not fill the delay slots here; you will confuse the --- register allocator. - -genCCall -    :: CmmCallTarget		-- function to call -    -> HintedCmmFormals		-- where to put the result -    -> HintedCmmActuals		-- arguments (of mixed type) -    -> NatM InstrBlock - --- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -ccallResultRegs =  - -genCCall fn cconv result_regs args -  = mapAccumLNat get_arg (allArgRegs, eXTRA_STK_ARGS_HERE) args -    	    	  	  `thenNat` \ ((unused,_), argCode) -> -    let -    	nRegs = length allArgRegs - length unused -    	code = asmSeqThen (map ($ []) argCode) -    in -    	returnSeq code [ -    	    LDA pv (AddrImm (ImmLab (ptext fn))), -    	    JSR ra (AddrReg pv) nRegs, -    	    LDGP gp (AddrReg ra)] -  where -    ------------------------ -    {-	Try to get a value into a specific register (or registers) for -	a call.  The first 6 arguments go into the appropriate -	argument register (separate registers for integer and floating -	point arguments, but used in lock-step), and the remaining -	arguments are dumped to the stack, beginning at 0(sp).  Our -	first argument is a pair of the list of remaining argument -	registers to be assigned for this call and the next stack -	offset to use for overflowing arguments.  This way, -	@get_Arg@ can be applied to all of a call's arguments using -	@mapAccumLNat@. -    -} -    get_arg -	:: ([(Reg,Reg)], Int)	-- Argument registers and stack offset (accumulator) -	-> StixTree		-- Current argument -	-> NatM (([(Reg,Reg)],Int), InstrBlock) -- Updated accumulator and code - -    -- We have to use up all of our argument registers first... - -    get_arg ((iDst,fDst):dsts, offset) arg -      = getRegister arg	    	    	    `thenNat` \ register -> -	let -	    reg  = if isFloatType pk then fDst else iDst -	    code = registerCode register reg -	    src  = registerName register reg -	    pk   = registerRep register -	in -	return ( -	    if isFloatType pk then -		((dsts, offset), if isFixed register then -		    code . mkSeqInstr (FMOV src fDst) -		    else code) -	    else -		((dsts, offset), if isFixed register then -		    code . mkSeqInstr (OR src (RIReg src) iDst) -		    else code)) - -    -- Once we have run out of argument registers, we move to the -    -- stack... - -    get_arg ([], offset) arg -      = getRegister arg			`thenNat` \ register -> -	getNewRegNat (registerRep register) -					`thenNat` \ tmp -> -	let -	    code = registerCode register tmp -	    src  = registerName register tmp -	    pk   = registerRep register -	    sz   = primRepToSize pk -	in -	return (([], offset + 1), code . mkSeqInstr (ST sz src (spRel offset))) - -trivialCode instr x (StInt y) -  | fits8Bits y -  = getRegister x		`thenNat` \ register -> -    getNewRegNat IntRep		`thenNat` \ tmp -> -    let -    	code = registerCode register tmp -    	src1 = registerName register tmp -    	src2 = ImmInt (fromInteger y) -    	code__2 dst = code . mkSeqInstr (instr src1 (RIImm src2) dst) -    in -    return (Any IntRep code__2) - -trivialCode instr x y -  = getRegister x		`thenNat` \ register1 -> -    getRegister y		`thenNat` \ register2 -> -    getNewRegNat IntRep		`thenNat` \ tmp1 -> -    getNewRegNat IntRep		`thenNat` \ tmp2 -> -    let -    	code1 = registerCode register1 tmp1 [] -    	src1  = registerName register1 tmp1 -    	code2 = registerCode register2 tmp2 [] -    	src2  = registerName register2 tmp2 -    	code__2 dst = asmSeqThen [code1, code2] . -    	    	     mkSeqInstr (instr src1 (RIReg src2) dst) -    in -    return (Any IntRep code__2) - ------------- -trivialUCode instr x -  = getRegister x		`thenNat` \ register -> -    getNewRegNat IntRep		`thenNat` \ tmp -> -    let -    	code = registerCode register tmp -    	src  = registerName register tmp -    	code__2 dst = code . mkSeqInstr (instr (RIReg src) dst) -    in -    return (Any IntRep code__2) - ------------- -trivialFCode _ instr x y -  = getRegister x		`thenNat` \ register1 -> -    getRegister y		`thenNat` \ register2 -> -    getNewRegNat FF64	`thenNat` \ tmp1 -> -    getNewRegNat FF64	`thenNat` \ tmp2 -> -    let -    	code1 = registerCode register1 tmp1 -    	src1  = registerName register1 tmp1 - -    	code2 = registerCode register2 tmp2 -    	src2  = registerName register2 tmp2 - -    	code__2 dst = asmSeqThen [code1 [], code2 []] . -    	    	      mkSeqInstr (instr src1 src2 dst) -    in -    return (Any FF64 code__2) - -trivialUFCode _ instr x -  = getRegister x		`thenNat` \ register -> -    getNewRegNat FF64	`thenNat` \ tmp -> -    let -    	code = registerCode register tmp -    	src  = registerName register tmp -    	code__2 dst = code . mkSeqInstr (instr src dst) -    in -    return (Any FF64 code__2) - -#if alpha_TARGET_ARCH - -coerceInt2FP _ x -  = getRegister x		`thenNat` \ register -> -    getNewRegNat IntRep		`thenNat` \ reg -> -    let -    	code = registerCode register reg -    	src  = registerName register reg - -    	code__2 dst = code . mkSeqInstrs [ -    	    ST Q src (spRel 0), -    	    LD TF dst (spRel 0), -    	    CVTxy Q TF dst dst] -    in -    return (Any FF64 code__2) - -------------- -coerceFP2Int x -  = getRegister x		`thenNat` \ register -> -    getNewRegNat FF64	`thenNat` \ tmp -> -    let -    	code = registerCode register tmp -    	src  = registerName register tmp - -    	code__2 dst = code . mkSeqInstrs [ -    	    CVTxy TF Q src tmp, -    	    ST TF tmp (spRel 0), -    	    LD Q dst (spRel 0)] -    in -    return (Any IntRep code__2) - -#endif /* alpha_TARGET_ARCH */ - - --} - - - - - diff --git a/compiler/nativeGen/Alpha/Instr.hs b/compiler/nativeGen/Alpha/Instr.hs deleted file mode 100644 index 990ea8bc1a..0000000000 --- a/compiler/nativeGen/Alpha/Instr.hs +++ /dev/null @@ -1,142 +0,0 @@ ------------------------------------------------------------------------------ --- --- Machine-dependent assembly language --- --- (c) The University of Glasgow 1993-2004 --- ------------------------------------------------------------------------------ - -#include "HsVersions.h" -#include "nativeGen/NCG.h" - -module Alpha.Instr ( ---	Cond(..), ---	Instr(..), ---	RI(..) -) - -where - -{- -import BlockId -import Regs -import Cmm -import FastString -import CLabel - -data Cond -	= ALWAYS	-- For BI (same as BR) -	| EQQ		-- For CMP and BI (NB: "EQ" is a 1.3 Prelude name) -	| GE		-- For BI only -	| GTT		-- For BI only (NB: "GT" is a 1.3 Prelude name) -	| LE		-- For CMP and BI -	| LTT		-- For CMP and BI (NB: "LT" is a 1.3 Prelude name) -	| NE		-- For BI only -	| NEVER		-- For BI (null instruction) -	| ULE		-- For CMP only -	| ULT		-- For CMP only -	deriving Eq -	 - --- ----------------------------------------------------------------------------- --- Machine's assembly language - --- We have a few common "instructions" (nearly all the pseudo-ops) but --- mostly all of 'Instr' is machine-specific. - --- Register or immediate -data RI  -	= RIReg Reg -	| RIImm Imm - -data Instr -	-- comment pseudo-op -	= COMMENT FastString		 - -	-- some static data spat out during code -	-- generation.  Will be extracted before -	-- pretty-printing. -	| LDATA   Section [CmmStatic]	 - -	-- start a new basic block.  Useful during -	-- codegen, removed later.  Preceding  -	-- instruction should be a jump, as per the -	-- invariants for a BasicBlock (see Cmm). -	| NEWBLOCK BlockId		 - -	-- specify current stack offset for -        -- benefit of subsequent passes -	| DELTA   Int - -	-- | spill this reg to a stack slot -	| SPILL   Reg Int - -	-- | reload this reg from a stack slot -	| RELOAD  Int Reg - -	-- Loads and stores. -	| LD	      Size Reg AddrMode		-- size, dst, src -	| LDA	      Reg AddrMode		-- dst, src -	| LDAH	      Reg AddrMode		-- dst, src -	| LDGP	      Reg AddrMode		-- dst, src -	| LDI	      Size Reg Imm     		-- size, dst, src -	| ST	      Size Reg AddrMode 	-- size, src, dst - -	-- Int Arithmetic. -	| CLR	      Reg		    	-- dst -	| ABS	      Size RI Reg	    	-- size, src, dst -	| NEG	      Size Bool RI Reg	   	-- size, overflow, src, dst -	| ADD	      Size Bool Reg RI Reg 	-- size, overflow, src, src, dst -	| SADD	      Size Size Reg RI Reg 	-- size, scale, src, src, dst -	| SUB	      Size Bool Reg RI Reg 	-- size, overflow, src, src, dst -	| SSUB	      Size Size Reg RI Reg 	-- size, scale, src, src, dst -	| MUL	      Size Bool Reg RI Reg 	-- size, overflow, src, src, dst -	| DIV	      Size Bool Reg RI Reg 	-- size, unsigned, src, src, dst -	| REM	      Size Bool Reg RI Reg 	-- size, unsigned, src, src, dst - -	-- Simple bit-twiddling. -	| NOT	      RI Reg -	| AND	      Reg RI Reg -	| ANDNOT      Reg RI Reg -	| OR	      Reg RI Reg -	| ORNOT	      Reg RI Reg -	| XOR	      Reg RI Reg -	| XORNOT      Reg RI Reg -	| SLL	      Reg RI Reg -	| SRL	      Reg RI Reg -	| SRA	      Reg RI Reg - -	| ZAP	      Reg RI Reg -	| ZAPNOT      Reg RI Reg - -	| NOP - -	-- Comparison -	| CMP	      Cond Reg RI Reg - -	-- Float Arithmetic. -	| FCLR	      Reg -	| FABS	      Reg Reg -	| FNEG	      Size Reg Reg -	| FADD	      Size Reg Reg Reg -	| FDIV	      Size Reg Reg Reg -	| FMUL	      Size Reg Reg Reg -	| FSUB	      Size Reg Reg Reg -	| CVTxy	      Size Size Reg Reg -	| FCMP	      Size Cond Reg Reg Reg -	| FMOV	      Reg Reg - -	-- Jumping around. -	| BI	      Cond Reg Imm -	| BF	      Cond Reg Imm -	| BR	      Imm -	| JMP	      Reg AddrMode Int -	| BSR	      Imm Int -	| JSR	      Reg AddrMode Int - -	-- Alpha-specific pseudo-ops. -	| FUNBEGIN CLabel -	| FUNEND CLabel - - --} diff --git a/compiler/nativeGen/Alpha/Ppr.hs-old b/compiler/nativeGen/Alpha/Ppr.hs-old deleted file mode 100644 index c14eef205d..0000000000 --- a/compiler/nativeGen/Alpha/Ppr.hs-old +++ /dev/null @@ -1,562 +0,0 @@ - -module Alpha.Ppr ( -{- -	pprReg, -	pprSize, -	pprCond, -	pprAddr, -	pprSectionHeader, -	pprTypeAndSizeDecl, -	pprRI, -	pprRegRIReg, -	pprSizeRegRegReg --} -) - -where - -{- -#include "nativeGen/NCG.h" -#include "HsVersions.h" - -import BlockId -import Cmm -import Regs		-- may differ per-platform -import Instrs - -import CLabel		( CLabel, pprCLabel, externallyVisibleCLabel, -			  labelDynamic, mkAsmTempLabel, entryLblToInfoLbl ) - -#if HAVE_SUBSECTIONS_VIA_SYMBOLS -import CLabel       ( mkDeadStripPreventer ) -#endif - -import Panic		( panic ) -import Unique		( pprUnique ) -import Pretty -import FastString -import qualified Outputable -import Outputable	( Outputable, pprPanic, ppr, docToSDoc) - -import Data.Array.ST -import Data.Word	( Word8 ) -import Control.Monad.ST -import Data.Char	( chr, ord ) -import Data.Maybe       ( isJust ) - - - -pprReg :: Reg -> Doc -pprReg r -  = case r of -      RealReg i      -> ppr_reg_no i -      VirtualRegI  u  -> text "%vI_"  <> asmSDoc (pprUnique u) -      VirtualRegHi u  -> text "%vHi_" <> asmSDoc (pprUnique u) -      VirtualRegF  u  -> text "%vF_"  <> asmSDoc (pprUnique u) -      VirtualRegD  u  -> text "%vD_"  <> asmSDoc (pprUnique u) -  where -    ppr_reg_no :: Int -> Doc -    ppr_reg_no i = ptext -      (case i of { -	 0 -> sLit "$0";    1 -> sLit "$1"; -	 2 -> sLit "$2";    3 -> sLit "$3"; -	 4 -> sLit "$4";    5 -> sLit "$5"; -	 6 -> sLit "$6";    7 -> sLit "$7"; -	 8 -> sLit "$8";    9 -> sLit "$9"; -	10 -> sLit "$10";  11 -> sLit "$11"; -	12 -> sLit "$12";  13 -> sLit "$13"; -	14 -> sLit "$14";  15 -> sLit "$15"; -	16 -> sLit "$16";  17 -> sLit "$17"; -	18 -> sLit "$18";  19 -> sLit "$19"; -	20 -> sLit "$20";  21 -> sLit "$21"; -	22 -> sLit "$22";  23 -> sLit "$23"; -	24 -> sLit "$24";  25 -> sLit "$25"; -	26 -> sLit "$26";  27 -> sLit "$27"; -	28 -> sLit "$28";  29 -> sLit "$29"; -	30 -> sLit "$30";  31 -> sLit "$31"; -	32 -> sLit "$f0";  33 -> sLit "$f1"; -	34 -> sLit "$f2";  35 -> sLit "$f3"; -	36 -> sLit "$f4";  37 -> sLit "$f5"; -	38 -> sLit "$f6";  39 -> sLit "$f7"; -	40 -> sLit "$f8";  41 -> sLit "$f9"; -	42 -> sLit "$f10"; 43 -> sLit "$f11"; -	44 -> sLit "$f12"; 45 -> sLit "$f13"; -	46 -> sLit "$f14"; 47 -> sLit "$f15"; -	48 -> sLit "$f16"; 49 -> sLit "$f17"; -	50 -> sLit "$f18"; 51 -> sLit "$f19"; -	52 -> sLit "$f20"; 53 -> sLit "$f21"; -	54 -> sLit "$f22"; 55 -> sLit "$f23"; -	56 -> sLit "$f24"; 57 -> sLit "$f25"; -	58 -> sLit "$f26"; 59 -> sLit "$f27"; -	60 -> sLit "$f28"; 61 -> sLit "$f29"; -	62 -> sLit "$f30"; 63 -> sLit "$f31"; -	_  -> sLit "very naughty alpha register" -      }) - - -pprSize :: Size -> Doc -pprSize x = ptext (case x of -	 B  -> sLit "b" -	 Bu -> sLit "bu" ---	 W  -> sLit "w" UNUSED ---	 Wu -> sLit "wu" UNUSED -	 L  -> sLit "l" -	 Q  -> sLit "q" ---	 FF -> sLit "f" UNUSED ---	 DF -> sLit "d" UNUSED ---	 GF -> sLit "g" UNUSED ---	 SF -> sLit "s" UNUSED -	 TF -> sLit "t" - - -pprCond :: Cond -> Doc -pprCond c  - = ptext (case c of -		EQQ  -> sLit "eq" -		LTT  -> sLit "lt" -		LE  -> sLit "le" -		ULT -> sLit "ult" -		ULE -> sLit "ule" -		NE  -> sLit "ne" -		GTT  -> sLit "gt" -		GE  -> sLit "ge") - - -pprAddr :: AddrMode -> Doc -pprAddr (AddrReg r) = parens (pprReg r) -pprAddr (AddrImm i) = pprImm i -pprAddr (AddrRegImm r1 i) -  = (<>) (pprImm i) (parens (pprReg r1)) - - -pprSectionHeader Text -    = ptext	(sLit "\t.text\n\t.align 3") - -pprSectionHeader Data -    = ptext	(sLit "\t.data\n\t.align 3") - -pprSectionHeader ReadOnlyData -    = ptext	(sLit "\t.data\n\t.align 3") - -pprSectionHeader RelocatableReadOnlyData -    = ptext	(sLit "\t.data\n\t.align 3") - -pprSectionHeader UninitialisedData -    = ptext	(sLit "\t.bss\n\t.align 3") - -pprSectionHeader ReadOnlyData16 -    = ptext	(sLit "\t.data\n\t.align 4") - -pprSectionHeader (OtherSection sec) -    = panic "PprMach.pprSectionHeader: unknown section" - - -pprTypeAndSizeDecl :: CLabel -> Doc -pprTypeAndSizeDecl lbl -  = empty - - - -pprInstr :: Instr -> Doc - -pprInstr (DELTA d) -   = pprInstr (COMMENT (mkFastString ("\tdelta = " ++ show d))) - -pprInstr (NEWBLOCK _) -   = panic "PprMach.pprInstr: NEWBLOCK" - -pprInstr (LDATA _ _) -   = panic "PprMach.pprInstr: LDATA" - -pprInstr (SPILL reg slot) -   = hcat [ -   	ptext (sLit "\tSPILL"), -	char '\t', -	pprReg reg, -	comma, -	ptext (sLit "SLOT") <> parens (int slot)] - -pprInstr (RELOAD slot reg) -   = hcat [ -   	ptext (sLit "\tRELOAD"), -	char '\t', -	ptext (sLit "SLOT") <> parens (int slot), -	comma, -	pprReg reg] - -pprInstr (LD size reg addr) -  = hcat [ -	ptext (sLit "\tld"), -	pprSize size, -	char '\t', -	pprReg reg, -	comma, -	pprAddr addr -    ] - -pprInstr (LDA reg addr) -  = hcat [ -	ptext (sLit "\tlda\t"), -	pprReg reg, -	comma, -	pprAddr addr -    ] - -pprInstr (LDAH reg addr) -  = hcat [ -	ptext (sLit "\tldah\t"), -	pprReg reg, -	comma, -	pprAddr addr -    ] - -pprInstr (LDGP reg addr) -  = hcat [ -	ptext (sLit "\tldgp\t"), -	pprReg reg, -	comma, -	pprAddr addr -    ] - -pprInstr (LDI size reg imm) -  = hcat [ -	ptext (sLit "\tldi"), -	pprSize size, -	char '\t', -	pprReg reg, -	comma, -	pprImm imm -    ] - -pprInstr (ST size reg addr) -  = hcat [ -	ptext (sLit "\tst"), -	pprSize size, -	char '\t', -	pprReg reg, -	comma, -	pprAddr addr -    ] - -pprInstr (CLR reg) -  = hcat [ -	ptext (sLit "\tclr\t"), -	pprReg reg -    ] - -pprInstr (ABS size ri reg) -  = hcat [ -	ptext (sLit "\tabs"), -	pprSize size, -	char '\t', -	pprRI ri, -	comma, -	pprReg reg -    ] - -pprInstr (NEG size ov ri reg) -  = hcat [ -	ptext (sLit "\tneg"), -	pprSize size, -	if ov then ptext (sLit "v\t") else char '\t', -	pprRI ri, -	comma, -	pprReg reg -    ] - -pprInstr (ADD size ov reg1 ri reg2) -  = hcat [ -	ptext (sLit "\tadd"), -	pprSize size, -	if ov then ptext (sLit "v\t") else char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (SADD size scale reg1 ri reg2) -  = hcat [ -	ptext (case scale of {{-UNUSED:L -> (sLit "\ts4");-} Q -> (sLit "\ts8")}), -	ptext (sLit "add"), -	pprSize size, -	char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (SUB size ov reg1 ri reg2) -  = hcat [ -	ptext (sLit "\tsub"), -	pprSize size, -	if ov then ptext (sLit "v\t") else char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (SSUB size scale reg1 ri reg2) -  = hcat [ -	ptext (case scale of {{-UNUSED:L -> (sLit "\ts4");-} Q -> (sLit "\ts8")}), -	ptext (sLit "sub"), -	pprSize size, -	char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (MUL size ov reg1 ri reg2) -  = hcat [ -	ptext (sLit "\tmul"), -	pprSize size, -	if ov then ptext (sLit "v\t") else char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (DIV size uns reg1 ri reg2) -  = hcat [ -	ptext (sLit "\tdiv"), -	pprSize size, -	if uns then ptext (sLit "u\t") else char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (REM size uns reg1 ri reg2) -  = hcat [ -	ptext (sLit "\trem"), -	pprSize size, -	if uns then ptext (sLit "u\t") else char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (NOT ri reg) -  = hcat [ -	ptext (sLit "\tnot"), -	char '\t', -	pprRI ri, -	comma, -	pprReg reg -    ] - -pprInstr (AND reg1 ri reg2) = pprRegRIReg (sLit "and") reg1 ri reg2 -pprInstr (ANDNOT reg1 ri reg2) = pprRegRIReg (sLit "andnot") reg1 ri reg2 -pprInstr (OR reg1 ri reg2) = pprRegRIReg (sLit "or") reg1 ri reg2 -pprInstr (ORNOT reg1 ri reg2) = pprRegRIReg (sLit "ornot") reg1 ri reg2 -pprInstr (XOR reg1 ri reg2) = pprRegRIReg (sLit "xor") reg1 ri reg2 -pprInstr (XORNOT reg1 ri reg2) = pprRegRIReg (sLit "xornot") reg1 ri reg2 - -pprInstr (SLL reg1 ri reg2) = pprRegRIReg (sLit "sll") reg1 ri reg2 -pprInstr (SRL reg1 ri reg2) = pprRegRIReg (sLit "srl") reg1 ri reg2 -pprInstr (SRA reg1 ri reg2) = pprRegRIReg (sLit "sra") reg1 ri reg2 - -pprInstr (ZAP reg1 ri reg2) = pprRegRIReg (sLit "zap") reg1 ri reg2 -pprInstr (ZAPNOT reg1 ri reg2) = pprRegRIReg (sLit "zapnot") reg1 ri reg2 - -pprInstr (NOP) = ptext (sLit "\tnop") - -pprInstr (CMP cond reg1 ri reg2) -  = hcat [ -	ptext (sLit "\tcmp"), -	pprCond cond, -	char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprInstr (FCLR reg) -  = hcat [ -	ptext (sLit "\tfclr\t"), -	pprReg reg -    ] - -pprInstr (FABS reg1 reg2) -  = hcat [ -	ptext (sLit "\tfabs\t"), -	pprReg reg1, -	comma, -	pprReg reg2 -    ] - -pprInstr (FNEG size reg1 reg2) -  = hcat [ -	ptext (sLit "\tneg"), -	pprSize size, -	char '\t', -	pprReg reg1, -	comma, -	pprReg reg2 -    ] - -pprInstr (FADD size reg1 reg2 reg3) = pprSizeRegRegReg (sLit "add") size reg1 reg2 reg3 -pprInstr (FDIV size reg1 reg2 reg3) = pprSizeRegRegReg (sLit "div") size reg1 reg2 reg3 -pprInstr (FMUL size reg1 reg2 reg3) = pprSizeRegRegReg (sLit "mul") size reg1 reg2 reg3 -pprInstr (FSUB size reg1 reg2 reg3) = pprSizeRegRegReg (sLit "sub") size reg1 reg2 reg3 - -pprInstr (CVTxy size1 size2 reg1 reg2) -  = hcat [ -	ptext (sLit "\tcvt"), -	pprSize size1, -	case size2 of {Q -> ptext (sLit "qc"); _ -> pprSize size2}, -	char '\t', -	pprReg reg1, -	comma, -	pprReg reg2 -    ] - -pprInstr (FCMP size cond reg1 reg2 reg3) -  = hcat [ -	ptext (sLit "\tcmp"), -	pprSize size, -	pprCond cond, -	char '\t', -	pprReg reg1, -	comma, -	pprReg reg2, -	comma, -	pprReg reg3 -    ] - -pprInstr (FMOV reg1 reg2) -  = hcat [ -	ptext (sLit "\tfmov\t"), -	pprReg reg1, -	comma, -	pprReg reg2 -    ] - -pprInstr (BI ALWAYS reg lab) = pprInstr (BR lab) - -pprInstr (BI NEVER reg lab) = empty - -pprInstr (BI cond reg lab) -  = hcat [ -	ptext (sLit "\tb"), -	pprCond cond, -	char '\t', -	pprReg reg, -	comma, -	pprImm lab -    ] - -pprInstr (BF cond reg lab) -  = hcat [ -	ptext (sLit "\tfb"), -	pprCond cond, -	char '\t', -	pprReg reg, -	comma, -	pprImm lab -    ] - -pprInstr (BR lab) -  = (<>) (ptext (sLit "\tbr\t")) (pprImm lab) - -pprInstr (JMP reg addr hint) -  = hcat [ -	ptext (sLit "\tjmp\t"), -	pprReg reg, -	comma, -	pprAddr addr, -	comma, -	int hint -    ] - -pprInstr (BSR imm n) -  = (<>) (ptext (sLit "\tbsr\t")) (pprImm imm) - -pprInstr (JSR reg addr n) -  = hcat [ -	ptext (sLit "\tjsr\t"), -	pprReg reg, -	comma, -	pprAddr addr -    ] - -pprInstr (FUNBEGIN clab) -  = hcat [ -	if (externallyVisibleCLabel clab) then -	    hcat [ptext (sLit "\t.globl\t"), pp_lab, char '\n'] -	else -	    empty, -	ptext (sLit "\t.ent "), -	pp_lab, -	char '\n', -	pp_lab, -	pp_ldgp, -	pp_lab, -	pp_frame -    ] -    where -	pp_lab = pprCLabel_asm clab - -        -- NEVER use commas within those string literals, cpp will ruin your day -	pp_ldgp  = hcat [ ptext (sLit ":\n\tldgp $29"), char ',', ptext (sLit "0($27)\n") ] -	pp_frame = hcat [ ptext (sLit "..ng:\n\t.frame $30"), char ',', -                          ptext (sLit "4240"), char ',', -                          ptext (sLit "$26"), char ',', -                          ptext (sLit "0\n\t.prologue 1") ] - -pprInstr (FUNEND clab) -  = (<>) (ptext (sLit "\t.align 4\n\t.end ")) (pprCLabel_asm clab) - - -pprRI :: RI -> Doc - -pprRI (RIReg r) = pprReg r -pprRI (RIImm r) = pprImm r - -pprRegRIReg :: LitString -> Reg -> RI -> Reg -> Doc -pprRegRIReg name reg1 ri reg2 -  = hcat [ - 	char '\t', -	ptext name, -	char '\t', -	pprReg reg1, -	comma, -	pprRI ri, -	comma, -	pprReg reg2 -    ] - -pprSizeRegRegReg :: LitString -> Size -> Reg -> Reg -> Reg -> Doc -pprSizeRegRegReg name size reg1 reg2 reg3 -  = hcat [ -	char '\t', -	ptext name, -	pprSize size, -	char '\t', -	pprReg reg1, -	comma, -	pprReg reg2, -	comma, -	pprReg reg3 -    ] - --} - - - diff --git a/compiler/nativeGen/Alpha/RegInfo.hs b/compiler/nativeGen/Alpha/RegInfo.hs deleted file mode 100644 index 7fdde4daf6..0000000000 --- a/compiler/nativeGen/Alpha/RegInfo.hs +++ /dev/null @@ -1,218 +0,0 @@ - ------------------------------------------------------------------------------ --- --- (c) The University of Glasgow 1996-2004 --- ------------------------------------------------------------------------------ - -module Alpha.RegInfo ( -{- -	RegUsage(..), -	noUsage, -	regUsage, -	patchRegs, -	jumpDests, -	isJumpish, -	patchJump, -	isRegRegMove, - -        JumpDest, canShortcut, shortcutJump, shortcutStatic, - -	maxSpillSlots, -	mkSpillInstr, -	mkLoadInstr, -	mkRegRegMoveInstr, -	mkBranchInstr --} -) - -where - -{- -#include "nativeGen/NCG.h" -#include "HsVersions.h" - - -import BlockId -import Cmm -import CLabel -import Instrs -import Regs -import Outputable -import Constants	( rESERVED_C_STACK_BYTES ) -import FastBool - -data RegUsage = RU [Reg] [Reg] - -noUsage :: RegUsage -noUsage  = RU [] [] - -regUsage :: Instr -> RegUsage - -regUsage instr = case instr of -    SPILL  reg slot	-> usage ([reg], []) -    RELOAD slot reg	-> usage ([], [reg]) -    LD B reg addr	-> usage (regAddr addr, [reg, t9]) -    LD Bu reg addr	-> usage (regAddr addr, [reg, t9]) ---  LD W reg addr	-> usage (regAddr addr, [reg, t9]) : UNUSED ---  LD Wu reg addr	-> usage (regAddr addr, [reg, t9]) : UNUSED -    LD sz reg addr	-> usage (regAddr addr, [reg]) -    LDA reg addr	-> usage (regAddr addr, [reg]) -    LDAH reg addr	-> usage (regAddr addr, [reg]) -    LDGP reg addr	-> usage (regAddr addr, [reg]) -    LDI sz reg imm	-> usage ([], [reg]) -    ST B reg addr	-> usage (reg : regAddr addr, [t9, t10]) ---  ST W reg addr	-> usage (reg : regAddr addr, [t9, t10]) : UNUSED -    ST sz reg addr	-> usage (reg : regAddr addr, []) -    CLR reg		-> usage ([], [reg]) -    ABS sz ri reg	-> usage (regRI ri, [reg]) -    NEG sz ov ri reg	-> usage (regRI ri, [reg]) -    ADD sz ov r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    SADD sz sc r1 ar r2 -> usage (r1 : regRI ar, [r2]) -    SUB sz ov r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    SSUB sz sc r1 ar r2 -> usage (r1 : regRI ar, [r2]) -    MUL sz ov r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    DIV sz un r1 ar r2	-> usage (r1 : regRI ar, [r2, t9, t10, t11, t12]) -    REM sz un r1 ar r2	-> usage (r1 : regRI ar, [r2, t9, t10, t11, t12]) -    NOT ri reg		-> usage (regRI ri, [reg]) -    AND r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    ANDNOT r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    OR r1 ar r2		-> usage (r1 : regRI ar, [r2]) -    ORNOT r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    XOR r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    XORNOT r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    SLL r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    SRL r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    SRA r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    ZAP r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    ZAPNOT r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    CMP co r1 ar r2	-> usage (r1 : regRI ar, [r2]) -    FCLR reg		-> usage ([], [reg]) -    FABS r1 r2		-> usage ([r1], [r2]) -    FNEG sz r1 r2	-> usage ([r1], [r2]) -    FADD sz r1 r2 r3	-> usage ([r1, r2], [r3]) -    FDIV sz r1 r2 r3	-> usage ([r1, r2], [r3]) -    FMUL sz r1 r2 r3	-> usage ([r1, r2], [r3]) -    FSUB sz r1 r2 r3	-> usage ([r1, r2], [r3]) -    CVTxy sz1 sz2 r1 r2 -> usage ([r1], [r2]) -    FCMP sz co r1 r2 r3 -> usage ([r1, r2], [r3]) -    FMOV r1 r2		-> usage ([r1], [r2]) - - -    -- We assume that all local jumps will be BI/BF/BR.	 JMP must be out-of-line. -    BI cond reg lbl	-> usage ([reg], []) -    BF cond reg lbl	-> usage ([reg], []) -    JMP reg addr hint	-> RU (mkRegSet (filter interesting (regAddr addr))) freeRegSet - -    BSR _ n		-> RU (argRegSet n) callClobberedRegSet -    JSR reg addr n	-> RU (argRegSet n) callClobberedRegSet - -    _			-> noUsage - -  where -    usage (src, dst) = RU (mkRegSet (filter interesting src)) -			  (mkRegSet (filter interesting dst)) - -    interesting (FixedReg _) = False -    interesting _ = True - -    regAddr (AddrReg r1)      = [r1] -    regAddr (AddrRegImm r1 _) = [r1] -    regAddr (AddrImm _)	      = [] - -    regRI (RIReg r) = [r] -    regRI  _	= [] - - -patchRegs :: Instr -> (Reg -> Reg) -> Instr -patchRegs instr env = case instr of -    SPILL  reg slot	-> SPILL (env reg) slot -    RELOAD slot reg	-> RELOAD slot (env reg) -    LD sz reg addr -> LD sz (env reg) (fixAddr addr) -    LDA reg addr -> LDA (env reg) (fixAddr addr) -    LDAH reg addr -> LDAH (env reg) (fixAddr addr) -    LDGP reg addr -> LDGP (env reg) (fixAddr addr) -    LDI sz reg imm -> LDI sz (env reg) imm -    ST sz reg addr -> ST sz (env reg) (fixAddr addr) -    CLR reg -> CLR (env reg) -    ABS sz ar reg -> ABS sz (fixRI ar) (env reg) -    NEG sz ov ar reg -> NEG sz ov (fixRI ar) (env reg) -    ADD sz ov r1 ar r2 -> ADD sz ov (env r1) (fixRI ar) (env r2) -    SADD sz sc r1 ar r2 -> SADD sz sc (env r1) (fixRI ar) (env r2) -    SUB sz ov r1 ar r2 -> SUB sz ov (env r1) (fixRI ar) (env r2) -    SSUB sz sc r1 ar r2 -> SSUB sz sc (env r1) (fixRI ar) (env r2) -    MUL sz ov r1 ar r2 -> MUL sz ov (env r1) (fixRI ar) (env r2) -    DIV sz un r1 ar r2 -> DIV sz un (env r1) (fixRI ar) (env r2) -    REM sz un r1 ar r2 -> REM sz un (env r1) (fixRI ar) (env r2) -    NOT ar reg -> NOT (fixRI ar) (env reg) -    AND r1 ar r2 -> AND (env r1) (fixRI ar) (env r2) -    ANDNOT r1 ar r2 -> ANDNOT (env r1) (fixRI ar) (env r2) -    OR r1 ar r2 -> OR (env r1) (fixRI ar) (env r2) -    ORNOT r1 ar r2 -> ORNOT (env r1) (fixRI ar) (env r2) -    XOR r1 ar r2 -> XOR (env r1) (fixRI ar) (env r2) -    XORNOT r1 ar r2 -> XORNOT (env r1) (fixRI ar) (env r2) -    SLL r1 ar r2 -> SLL (env r1) (fixRI ar) (env r2) -    SRL r1 ar r2 -> SRL (env r1) (fixRI ar) (env r2) -    SRA r1 ar r2 -> SRA (env r1) (fixRI ar) (env r2) -    ZAP r1 ar r2 -> ZAP (env r1) (fixRI ar) (env r2) -    ZAPNOT r1 ar r2 -> ZAPNOT (env r1) (fixRI ar) (env r2) -    CMP co r1 ar r2 -> CMP co (env r1) (fixRI ar) (env r2) -    FCLR reg -> FCLR (env reg) -    FABS r1 r2 -> FABS (env r1) (env r2) -    FNEG s r1 r2 -> FNEG s (env r1) (env r2) -    FADD s r1 r2 r3 -> FADD s (env r1) (env r2) (env r3) -    FDIV s r1 r2 r3 -> FDIV s (env r1) (env r2) (env r3) -    FMUL s r1 r2 r3 -> FMUL s (env r1) (env r2) (env r3) -    FSUB s r1 r2 r3 -> FSUB s (env r1) (env r2) (env r3) -    CVTxy s1 s2 r1 r2 -> CVTxy s1 s2 (env r1) (env r2) -    FCMP s co r1 r2 r3 -> FCMP s co (env r1) (env r2) (env r3) -    FMOV r1 r2 -> FMOV (env r1) (env r2) -    BI cond reg lbl -> BI cond (env reg) lbl -    BF cond reg lbl -> BF cond (env reg) lbl -    JMP reg addr hint -> JMP (env reg) (fixAddr addr) hint -    JSR reg addr i -> JSR (env reg) (fixAddr addr) i -    _ -> instr -  where -    fixAddr (AddrReg r1)       = AddrReg (env r1) -    fixAddr (AddrRegImm r1 i)  = AddrRegImm (env r1) i -    fixAddr other	       = other - -    fixRI (RIReg r) = RIReg (env r) -    fixRI other	= other - - -mkSpillInstr -   :: Reg		-- register to spill -   -> Int		-- current stack delta -   -> Int		-- spill slot to use -   -> Instr - -mkSpillInstr reg delta slot -  = let	off     = spillSlotToOffset slot -    in -    -- Alpha: spill below the stack pointer (?) -    ST sz dyn (spRel (- (off `div` 8))) - - -mkLoadInstr -   :: Reg		-- register to load -   -> Int		-- current stack delta -   -> Int		-- spill slot to use -   -> Instr -mkLoadInstr reg delta slot -  = let off     = spillSlotToOffset slot -    in -	 LD  sz dyn (spRel (- (off `div` 8))) - - -mkBranchInstr -    :: BlockId -    -> [Instr] - -mkBranchInstr id = [BR id] - --} - - - - diff --git a/compiler/nativeGen/Alpha/Regs.hs b/compiler/nativeGen/Alpha/Regs.hs deleted file mode 100644 index ee490509de..0000000000 --- a/compiler/nativeGen/Alpha/Regs.hs +++ /dev/null @@ -1,323 +0,0 @@ --- ----------------------------------------------------------------------------- --- --- (c) The University of Glasgow 1994-2004 ---  --- Alpha support is rotted and incomplete. --- ----------------------------------------------------------------------------- - - -module Alpha.Regs ( -{- -	Size(..), -	AddrMode(..), -	fits8Bits, -	fReg, -	gp, pv, ra, sp, t9, t10, t11, t12, v0, f0, zeroh --} -) - -where - -{- -#include "nativeGen/NCG.h" -#include "HsVersions.h" -#include "../includes/stg/MachRegs.h" - -import RegsBase - -import BlockId -import Cmm -import CLabel           ( CLabel, mkMainCapabilityLabel ) -import Pretty -import Outputable	( Outputable(..), pprPanic, panic ) -import qualified Outputable -import Unique -import UniqSet -import Constants -import FastTypes -import FastBool -import UniqFM - - -data Size -	= B	    -- byte -	| Bu ---	| W	    -- word (2 bytes): UNUSED ---	| Wu    -- : UNUSED -	| L	    -- longword (4 bytes) -	| Q	    -- quadword (8 bytes) ---	| FF    -- VAX F-style floating pt: UNUSED ---	| GF    -- VAX G-style floating pt: UNUSED ---	| DF    -- VAX D-style floating pt: UNUSED ---	| SF    -- IEEE single-precision floating pt: UNUSED -	| TF    -- IEEE double-precision floating pt -	deriving Eq - - -data AddrMode -	= AddrImm	Imm -	| AddrReg	Reg -	| AddrRegImm	Reg Imm - - -addrOffset :: AddrMode -> Int -> Maybe AddrMode -addrOffset addr off -  = case addr of -      _ -> panic "MachMisc.addrOffset not defined for Alpha" - -fits8Bits :: Integer -> Bool -fits8Bits i = i >= -256 && i < 256 - - --- The Alpha has 64 registers of interest; 32 integer registers and 32 floating --- point registers.  The mapping of STG registers to alpha machine registers --- is defined in StgRegs.h.  We are, of course, prepared for any eventuality. - -fReg :: Int -> RegNo -fReg x = (32 + x) - -v0, f0, ra, pv, gp, sp, zeroh :: Reg -v0    = realReg 0 -f0    = realReg (fReg 0) -ra    = FixedReg ILIT(26) -pv    = t12 -gp    = FixedReg ILIT(29) -sp    = FixedReg ILIT(30) -zeroh = FixedReg ILIT(31) -- "zero" is used in 1.3 (MonadZero method) - -t9, t10, t11, t12 :: Reg -t9  = realReg 23 -t10 = realReg 24 -t11 = realReg 25 -t12 = realReg 27 - - -#define f0 32 -#define f1 33 -#define f2 34 -#define f3 35 -#define f4 36 -#define f5 37 -#define f6 38 -#define f7 39 -#define f8 40 -#define f9 41 -#define f10 42 -#define f11 43 -#define f12 44 -#define f13 45 -#define f14 46 -#define f15 47 -#define f16 48 -#define f17 49 -#define f18 50 -#define f19 51 -#define f20 52 -#define f21 53 -#define f22 54 -#define f23 55 -#define f24 56 -#define f25 57 -#define f26 58 -#define f27 59 -#define f28 60 -#define f29 61 -#define f30 62 -#define f31 63 - - --- allMachRegs is the complete set of machine regs. -allMachRegNos :: [RegNo] -allMachRegNos	= [0..63] - - --- these are the regs which we cannot assume stay alive over a --- C call.   -callClobberedRegs :: [Reg] -callClobberedRegs - =	[0, 1, 2, 3, 4, 5, 6, 7, 8, -	 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, -	 fReg 0, fReg 1, fReg 10, fReg 11, fReg 12, fReg 13, fReg 14, fReg 15, -	 fReg 16, fReg 17, fReg 18, fReg 19, fReg 20, fReg 21, fReg 22, fReg 23, -	 fReg 24, fReg 25, fReg 26, fReg 27, fReg 28, fReg 29, fReg 30] - - --- argRegs is the set of regs which are read for an n-argument call to C. --- For archs which pass all args on the stack (x86), is empty. --- Sparc passes up to the first 6 args in regs. --- Dunno about Alpha. -argRegs :: RegNo -> [Reg] - -argRegs 0 = [] -argRegs 1 = freeMappedRegs [16, fReg 16] -argRegs 2 = freeMappedRegs [16, 17, fReg 16, fReg 17] -argRegs 3 = freeMappedRegs [16, 17, 18, fReg 16, fReg 17, fReg 18] -argRegs 4 = freeMappedRegs [16, 17, 18, 19, fReg 16, fReg 17, fReg 18, fReg 19] -argRegs 5 = freeMappedRegs [16, 17, 18, 19, 20, fReg 16, fReg 17, fReg 18, fReg 19, fReg 20] -argRegs 6 = freeMappedRegs [16, 17, 18, 19, 20, 21, fReg 16, fReg 17, fReg 18, fReg 19, fReg 20, fReg 21] -argRegs _ = panic "MachRegs.argRegs(alpha): don't know about >6 arguments!" - - --- all of the arg regs ?? -allArgRegs :: [(Reg, Reg)] -allArgRegs = [(realReg i, realReg (fReg i)) | i <- [16..21]] - - --- horror show ----------------------------------------------------------------- - -freeReg :: RegNo -> FastBool - -freeReg 26 = fastBool False  -- return address (ra) -freeReg 28 = fastBool False  -- reserved for the assembler (at) -freeReg 29 = fastBool False  -- global pointer (gp) -freeReg 30 = fastBool False  -- stack pointer (sp) -freeReg 31 = fastBool False  -- always zero (zeroh) -freeReg 63 = fastBool False  -- always zero (f31) - -#ifdef REG_Base -freeReg REG_Base = fastBool False -#endif -#ifdef REG_R1 -freeReg REG_R1   = fastBool False -#endif	 -#ifdef REG_R2   -freeReg REG_R2   = fastBool False -#endif	 -#ifdef REG_R3   -freeReg REG_R3   = fastBool False -#endif	 -#ifdef REG_R4   -freeReg REG_R4   = fastBool False -#endif	 -#ifdef REG_R5   -freeReg REG_R5   = fastBool False -#endif	 -#ifdef REG_R6   -freeReg REG_R6   = fastBool False -#endif	 -#ifdef REG_R7   -freeReg REG_R7   = fastBool False -#endif	 -#ifdef REG_R8   -freeReg REG_R8   = fastBool False -#endif -#ifdef REG_F1 -freeReg REG_F1 = fastBool False -#endif -#ifdef REG_F2 -freeReg REG_F2 = fastBool False -#endif -#ifdef REG_F3 -freeReg REG_F3 = fastBool False -#endif -#ifdef REG_F4 -freeReg REG_F4 = fastBool False -#endif -#ifdef REG_D1 -freeReg REG_D1 = fastBool False -#endif -#ifdef REG_D2 -freeReg REG_D2 = fastBool False -#endif -#ifdef REG_Sp  -freeReg REG_Sp   = fastBool False -#endif  -#ifdef REG_Su -freeReg REG_Su   = fastBool False -#endif  -#ifdef REG_SpLim  -freeReg REG_SpLim = fastBool False -#endif  -#ifdef REG_Hp  -freeReg REG_Hp   = fastBool False -#endif -#ifdef REG_HpLim -freeReg REG_HpLim = fastBool False -#endif -freeReg n               = fastBool True - - ---  | Returns 'Nothing' if this global register is not stored --- in a real machine register, otherwise returns @'Just' reg@, where --- reg is the machine register it is stored in. - -globalRegMaybe :: GlobalReg -> Maybe Reg - -#ifdef REG_Base -globalRegMaybe BaseReg			= Just (RealReg REG_Base) -#endif -#ifdef REG_R1 -globalRegMaybe (VanillaReg 1 _)		= Just (RealReg REG_R1) -#endif  -#ifdef REG_R2  -globalRegMaybe (VanillaReg 2 _)		= Just (RealReg REG_R2) -#endif  -#ifdef REG_R3  -globalRegMaybe (VanillaReg 3 _) 	= Just (RealReg REG_R3) -#endif  -#ifdef REG_R4  -globalRegMaybe (VanillaReg 4 _)		= Just (RealReg REG_R4) -#endif  -#ifdef REG_R5  -globalRegMaybe (VanillaReg 5 _)		= Just (RealReg REG_R5) -#endif  -#ifdef REG_R6  -globalRegMaybe (VanillaReg 6 _)		= Just (RealReg REG_R6) -#endif  -#ifdef REG_R7  -globalRegMaybe (VanillaReg 7 _)		= Just (RealReg REG_R7) -#endif  -#ifdef REG_R8  -globalRegMaybe (VanillaReg 8 _)		= Just (RealReg REG_R8) -#endif -#ifdef REG_R9  -globalRegMaybe (VanillaReg 9 _)		= Just (RealReg REG_R9) -#endif -#ifdef REG_R10  -globalRegMaybe (VanillaReg 10 _)	= Just (RealReg REG_R10) -#endif -#ifdef REG_F1 -globalRegMaybe (FloatReg 1)		= Just (RealReg REG_F1) -#endif				 	 -#ifdef REG_F2			 	 -globalRegMaybe (FloatReg 2)		= Just (RealReg REG_F2) -#endif				 	 -#ifdef REG_F3			 	 -globalRegMaybe (FloatReg 3)		= Just (RealReg REG_F3) -#endif				 	 -#ifdef REG_F4			 	 -globalRegMaybe (FloatReg 4)		= Just (RealReg REG_F4) -#endif				 	 -#ifdef REG_D1			 	 -globalRegMaybe (DoubleReg 1)		= Just (RealReg REG_D1) -#endif				 	 -#ifdef REG_D2			 	 -globalRegMaybe (DoubleReg 2)		= Just (RealReg REG_D2) -#endif -#ifdef REG_Sp	     -globalRegMaybe Sp		   	= Just (RealReg REG_Sp) -#endif -#ifdef REG_Lng1			 	 -globalRegMaybe (LongReg 1)		= Just (RealReg REG_Lng1) -#endif				 	 -#ifdef REG_Lng2			 	 -globalRegMaybe (LongReg 2)		= Just (RealReg REG_Lng2) -#endif -#ifdef REG_SpLim	    			 -globalRegMaybe SpLim		   	= Just (RealReg REG_SpLim) -#endif	    				 -#ifdef REG_Hp	   			 -globalRegMaybe Hp		   	= Just (RealReg REG_Hp) -#endif	    				 -#ifdef REG_HpLim      			 -globalRegMaybe HpLim		   	= Just (RealReg REG_HpLim) -#endif	    				 -#ifdef REG_CurrentTSO      			 -globalRegMaybe CurrentTSO	   	= Just (RealReg REG_CurrentTSO) -#endif	    				 -#ifdef REG_CurrentNursery      			 -globalRegMaybe CurrentNursery	   	= Just (RealReg REG_CurrentNursery) -#endif	    				 -globalRegMaybe _		   	= Nothing - --} diff --git a/compiler/nativeGen/AsmCodeGen.lhs b/compiler/nativeGen/AsmCodeGen.lhs index 767dc99f61..27858dc847 100644 --- a/compiler/nativeGen/AsmCodeGen.lhs +++ b/compiler/nativeGen/AsmCodeGen.lhs @@ -13,13 +13,7 @@ module AsmCodeGen ( nativeCodeGen ) where  #include "nativeGen/NCG.h" -#if   alpha_TARGET_ARCH -import Alpha.CodeGen -import Alpha.Regs -import Alpha.RegInfo -import Alpha.Instr - -#elif i386_TARGET_ARCH || x86_64_TARGET_ARCH +#if i386_TARGET_ARCH || x86_64_TARGET_ARCH  import X86.CodeGen  import X86.Regs  import X86.Instr @@ -64,7 +58,7 @@ import NCGMonad  import BlockId  import CgUtils		( fixStgRegisters )  import OldCmm -import CmmOpt		( cmmMiniInline, cmmMachOpFold ) +import CmmOpt		( cmmEliminateDeadBlocks, cmmMiniInline, cmmMachOpFold )  import OldPprCmm  import CLabel @@ -378,10 +372,25 @@ cmmNativeGen dflags us cmm count  			, Nothing  			, mPprStats) -	---- generate jump tables +        ---- x86fp_kludge.  This pass inserts ffree instructions to clear +        ---- the FPU stack on x86.  The x86 ABI requires that the FPU stack +        ---- is clear, and library functions can return odd results if it +        ---- isn't. +        ---- +        ---- NB. must happen before shortcutBranches, because that +        ---- generates JXX_GBLs which we can't fix up in x86fp_kludge. +        let kludged = +#if i386_TARGET_ARCH +	 	{-# SCC "x86fp_kludge" #-} +                map x86fp_kludge alloced +#else +                alloced +#endif + +        ---- generate jump tables  	let tabled	=  		{-# SCC "generateJumpTables" #-} -		alloced ++ generateJumpTables alloced +                generateJumpTables kludged  	---- shortcut branches  	let shorted	= @@ -393,27 +402,18 @@ cmmNativeGen dflags us cmm count  	 	{-# SCC "sequenceBlocks" #-}  	 	map sequenceTop shorted -	---- x86fp_kludge -	let kludged = -#if i386_TARGET_ARCH -	 	{-# SCC "x86fp_kludge" #-} -	 	map x86fp_kludge sequenced -#else -		sequenced -#endif - -	---- expansion of SPARC synthetic instrs +        ---- expansion of SPARC synthetic instrs  #if sparc_TARGET_ARCH  	let expanded =   		{-# SCC "sparc_expand" #-} -		map expandTop kludged +                map expandTop sequenced  	dumpIfSet_dyn dflags  		Opt_D_dump_asm_expanded "Synthetic instructions expanded"  		(vcat $ map (docToSDoc . pprNatCmmTop) expanded)  #else  	let expanded =  -		kludged +                sequenced  #endif  	return 	( usAlloc @@ -621,8 +621,8 @@ makeFarBranches = id  generateJumpTables  	:: [NatCmmTop Instr] -> [NatCmmTop Instr]  generateJumpTables xs = concatMap f xs -    where f (CmmProc _ _ (ListGraph xs)) = concatMap g xs -          f _ = [] +    where f p@(CmmProc _ _ (ListGraph xs)) = p : concatMap g xs +          f p = [p]            g (BasicBlock _ xs) = catMaybes (map generateJumpTableForInstr xs)  -- ----------------------------------------------------------------------------- @@ -735,10 +735,9 @@ Here we do:               and position independent refs          (ii) compile a list of imported symbols -Ideas for other things we could do (ToDo): +Ideas for other things we could do:    - shortcut jumps-to-jumps -  - eliminate dead code blocks    - simple CSE: if an expr is assigned to a temp, then replace later occs of      that expr with the temp, until the expr is no longer valid (can push through      temp assignments, and certain assigns to mem...) @@ -747,7 +746,7 @@ Ideas for other things we could do (ToDo):  cmmToCmm :: DynFlags -> RawCmmTop -> (RawCmmTop, [CLabel])  cmmToCmm _ top@(CmmData _ _) = (top, [])  cmmToCmm dflags (CmmProc info lbl (ListGraph blocks)) = runCmmOpt dflags $ do -  blocks' <- mapM cmmBlockConFold (cmmMiniInline blocks) +  blocks' <- mapM cmmBlockConFold (cmmMiniInline (cmmEliminateDeadBlocks blocks))    return $ CmmProc info lbl (ListGraph blocks')  newtype CmmOptM a = CmmOptM (([CLabel], DynFlags) -> (# a, [CLabel] #)) diff --git a/compiler/nativeGen/PPC/Regs.hs b/compiler/nativeGen/PPC/Regs.hs index 73e0c2023e..7a2a84b68c 100644 --- a/compiler/nativeGen/PPC/Regs.hs +++ b/compiler/nativeGen/PPC/Regs.hs @@ -209,7 +209,6 @@ spRel n	= AddrRegImm sp (ImmInt (n * wORD_SIZE))  -- argRegs is the set of regs which are read for an n-argument call to C.  -- For archs which pass all args on the stack (x86), is empty.  -- Sparc passes up to the first 6 args in regs. --- Dunno about Alpha.  argRegs :: RegNo -> [Reg]  argRegs 0 = []  argRegs 1 = map regSingle [3] diff --git a/compiler/nativeGen/Platform.hs b/compiler/nativeGen/Platform.hs index 20cb5f5e96..7b2502d96e 100644 --- a/compiler/nativeGen/Platform.hs +++ b/compiler/nativeGen/Platform.hs @@ -31,8 +31,7 @@ data Platform  --	about what instruction set extensions an architecture might support.  --  data Arch -	= ArchAlpha -	| ArchX86 +	= ArchX86  	| ArchX86_64  	| ArchPPC  	| ArchPPC_64 @@ -70,9 +69,7 @@ defaultTargetPlatform  -- | Move the evil TARGET_ARCH #ifdefs into Haskell land.  defaultTargetArch :: Arch -#if   alpha_TARGET_ARCH -defaultTargetArch	= ArchAlpha -#elif i386_TARGET_ARCH +#if i386_TARGET_ARCH  defaultTargetArch	= ArchX86  #elif x86_64_TARGET_ARCH  defaultTargetArch	= ArchX86_64 diff --git a/compiler/nativeGen/X86/Instr.hs b/compiler/nativeGen/X86/Instr.hs index e934a6d4ef..92655d1693 100644 --- a/compiler/nativeGen/X86/Instr.hs +++ b/compiler/nativeGen/X86/Instr.hs @@ -746,7 +746,7 @@ i386_insert_ffrees blocks       where p insn r = case insn of                          CALL _ _ -> GFREE : insn : r                          JMP _    -> GFREE : insn : r -                        JXX_GBL _ _ -> GFREE : insn : r +                        JXX_GBL _ _ -> panic "i386_insert_ffrees: cannot handle JXX_GBL"                          _        -> insn : r  -- if you ever add a new FP insn to the fake x86 FP insn set, diff --git a/compiler/nativeGen/X86/Regs.hs b/compiler/nativeGen/X86/Regs.hs index 094b74dc37..28d148c12c 100644 --- a/compiler/nativeGen/X86/Regs.hs +++ b/compiler/nativeGen/X86/Regs.hs @@ -249,7 +249,6 @@ floatregnos = fakeregnos ++ xmmregnos;  -- argRegs is the set of regs which are read for an n-argument call to C.  -- For archs which pass all args on the stack (x86), is empty.  -- Sparc passes up to the first 6 args in regs. --- Dunno about Alpha.  argRegs :: RegNo -> [Reg]  argRegs _	= panic "MachRegs.argRegs(x86): should not be used!" @@ -333,10 +332,24 @@ fake5 = regSingle 21  {-  AMD x86_64 architecture: -- Registers 0-16 have 32-bit counterparts (eax, ebx etc.) -- Registers 0-7 have 16-bit counterparts (ax, bx etc.) -- Registers 0-3 have 8 bit counterparts (ah, bh etc.) - +- All 16 integer registers are addressable as 8, 16, 32 and 64-bit values: + +  8     16    32    64 +  --------------------- +  al    ax    eax   rax +  bl    bx    ebx   rbx +  cl    cx    ecx   rcx +  dl    dx    edx   rdx +  sil   si    esi   rsi +  dil   si    edi   rdi +  bpl   bp    ebp   rbp +  spl   sp    esp   rsp +  r10b  r10w  r10d  r10 +  r11b  r11w  r11d  r11 +  r12b  r12w  r12d  r12 +  r13b  r13w  r13d  r13 +  r14b  r14w  r14d  r14 +  r15b  r15w  r15d  r15  -}  rax, rbx, rcx, rdx, rsp, rbp, rsi, rdi,  diff --git a/compiler/parser/Lexer.x b/compiler/parser/Lexer.x index a2d2276901..46f7488dcc 100644 --- a/compiler/parser/Lexer.x +++ b/compiler/parser/Lexer.x @@ -1893,6 +1893,7 @@ mkPState flags buf loc =                 .|. unboxedTuplesBit  `setBitIf` xopt Opt_UnboxedTuples   flags                 .|. datatypeContextsBit `setBitIf` xopt Opt_DatatypeContexts flags                 .|. transformComprehensionsBit `setBitIf` xopt Opt_TransformListComp flags +               .|. transformComprehensionsBit `setBitIf` xopt Opt_MonadComprehensions flags                 .|. rawTokenStreamBit `setBitIf` dopt Opt_KeepRawTokenStream flags                 .|. alternativeLayoutRuleBit `setBitIf` xopt Opt_AlternativeLayoutRule flags                 .|. relaxedLayoutBit  `setBitIf` xopt Opt_RelaxedLayout flags diff --git a/compiler/parser/Parser.y.pp b/compiler/parser/Parser.y.pp index bfadfbaff8..aa20ea6799 100644 --- a/compiler/parser/Parser.y.pp +++ b/compiler/parser/Parser.y.pp @@ -1283,14 +1283,9 @@ exp10 :: { LHsExpr RdrName }     	| 'case' exp 'of' altslist		{ LL $ HsCase $2 (mkMatchGroup (unLoc $4)) }  	| '-' fexp				{ LL $ NegApp $2 noSyntaxExpr } -  	| 'do' stmtlist			{% let loc = comb2 $1 $2 in -					   checkDo loc (unLoc $2)  >>= \ (stmts,body) -> -					   return (L loc (mkHsDo DoExpr stmts body)) } -  	| 'mdo' stmtlist		{% let loc = comb2 $1 $2 in -					   checkDo loc (unLoc $2)  >>= \ (stmts,body) -> -                                           return (L loc (mkHsDo MDoExpr -                                                                 [L loc (mkRecStmt stmts)] -                                                                 body)) } +  	| 'do' stmtlist			{ L (comb2 $1 $2) (mkHsDo DoExpr  (unLoc $2)) } +  	| 'mdo' stmtlist		{ L (comb2 $1 $2) (mkHsDo MDoExpr (unLoc $2)) } +          | scc_annot exp		    		{ LL $ if opt_SccProfilingOn  							then HsSCC (unLoc $1) $2  							else HsPar $2 } @@ -1465,7 +1460,10 @@ list :: { LHsExpr RdrName }  	| texp ',' exp '..' 	{ LL $ ArithSeq noPostTcExpr (FromThen $1 $3) }  	| texp '..' exp	 	{ LL $ ArithSeq noPostTcExpr (FromTo $1 $3) }  	| texp ',' exp '..' exp	{ LL $ ArithSeq noPostTcExpr (FromThenTo $1 $3 $5) } -	| texp '|' flattenedpquals	{ sL (comb2 $1 $>) $ mkHsDo ListComp (unLoc $3) $1 } +	| texp '|' flattenedpquals	 +             {% checkMonadComp >>= \ ctxt -> +		return (sL (comb2 $1 $>) $  +                        mkHsComp ctxt (unLoc $3) $1) }  lexps :: { Located [LHsExpr RdrName] }  	: lexps ',' texp 		{ LL (((:) $! $3) $! unLoc $1) } @@ -1480,7 +1478,7 @@ flattenedpquals :: { Located [LStmt RdrName] }                      -- We just had one thing in our "parallel" list so                       -- we simply return that thing directly -                    qss -> L1 [L1 $ ParStmt [(qs, undefined) | qs <- qss]] +                    qss -> L1 [L1 $ ParStmt [(qs, undefined) | qs <- qss] noSyntaxExpr noSyntaxExpr noSyntaxExpr]                      -- We actually found some actual parallel lists so                      -- we wrap them into as a ParStmt                  } @@ -1537,7 +1535,7 @@ parr :: { LHsExpr RdrName }  						       (reverse (unLoc $1)) }  	| texp '..' exp	 		{ LL $ PArrSeq noPostTcExpr (FromTo $1 $3) }  	| texp ',' exp '..' exp		{ LL $ PArrSeq noPostTcExpr (FromThenTo $1 $3 $5) } -	| texp '|' flattenedpquals	{ LL $ mkHsDo PArrComp (unLoc $3) $1 } +	| texp '|' flattenedpquals	{ LL $ mkHsComp PArrComp (unLoc $3) $1 }  -- We are reusing `lexps' and `flattenedpquals' from the list case. diff --git a/compiler/parser/RdrHsSyn.lhs b/compiler/parser/RdrHsSyn.lhs index 47abf232e2..3b14990ec0 100644 --- a/compiler/parser/RdrHsSyn.lhs +++ b/compiler/parser/RdrHsSyn.lhs @@ -40,8 +40,7 @@ module RdrHsSyn (  	checkPattern,	      -- HsExp -> P HsPat  	bang_RDR,  	checkPatterns,	      -- SrcLoc -> [HsExp] -> P [HsPat] -	checkDo,	      -- [Stmt] -> P [Stmt] -	checkMDo,	      -- [Stmt] -> P [Stmt] +	checkMonadComp,       -- P (HsStmtContext RdrName)  	checkValDef,	      -- (SrcLoc, HsExp, HsRhs, [HsDecl]) -> P HsDecl  	checkValSig,	      -- (SrcLoc, HsExp, HsRhs, [HsDecl]) -> P HsDecl  	checkDoAndIfThenElse, @@ -54,6 +53,7 @@ import Class            ( FunDep )  import TypeRep          ( Kind )  import RdrName		( RdrName, isRdrTyVar, isRdrTc, mkUnqual, rdrNameOcc,   			  isRdrDataCon, isUnqual, getRdrName, setRdrNameSpace ) +import Name             ( Name )  import BasicTypes	( maxPrecedence, Activation(..), RuleMatchInfo,                            InlinePragma(..), InlineSpec(..) )  import Lexer @@ -611,34 +611,6 @@ checkPred (L spn ty)      check loc _                        _    = parseErrorSDoc loc                                  (text "malformed class assertion:" <+> ppr ty) ---------------------------------------------------------------------------- --- Checking statements in a do-expression --- 	We parse   do { e1 ; e2 ; } --- 	as [ExprStmt e1, ExprStmt e2] --- checkDo (a) checks that the last thing is an ExprStmt ---	   (b) returns it separately --- same comments apply for mdo as well - -checkDo, checkMDo :: SrcSpan -> [LStmt RdrName] -> P ([LStmt RdrName], LHsExpr RdrName) - -checkDo	 = checkDoMDo "a " "'do'" -checkMDo = checkDoMDo "an " "'mdo'" - -checkDoMDo :: String -> String -> SrcSpan -> [LStmt RdrName] -> P ([LStmt RdrName], LHsExpr RdrName) -checkDoMDo _   nm loc []   = parseErrorSDoc loc (text ("Empty " ++ nm ++ " construct")) -checkDoMDo pre nm _   ss   = do -  check ss -  where  -	check  []                     = panic "RdrHsSyn:checkDoMDo" -	check  [L _ (ExprStmt e _ _)] = return ([], e) -	check  [L l e] = parseErrorSDoc l -                         (text ("The last statement in " ++ pre ++ nm ++ -					            " construct must be an expression:") -                       $$ ppr e) -	check (s:ss) = do -	  (ss',e') <-  check ss -	  return ((s:ss'),e') -  -- -------------------------------------------------------------------------  -- Checking Patterns. @@ -912,6 +884,20 @@ isFunLhs e = go e []  		 _ -> return Nothing }     go _ _ = return Nothing + +--------------------------------------------------------------------------- +-- Check for monad comprehensions +-- +-- If the flag MonadComprehensions is set, return a `MonadComp' context, +-- otherwise use the usual `ListComp' context + +checkMonadComp :: P (HsStmtContext Name) +checkMonadComp = do +    pState <- getPState +    return $ if xopt Opt_MonadComprehensions (dflags pState) +                then MonadComp +                else ListComp +  ---------------------------------------------------------------------------  -- Miscellaneous utilities diff --git a/compiler/prelude/PrelNames.lhs b/compiler/prelude/PrelNames.lhs index b7396a7233..99221e3f17 100644 --- a/compiler/prelude/PrelNames.lhs +++ b/compiler/prelude/PrelNames.lhs @@ -160,6 +160,7 @@ basicKnownKeyNames  	-- Monad stuff  	thenIOName, bindIOName, returnIOName, failIOName,  	failMName, bindMName, thenMName, returnMName, +        fmapName,  	-- MonadRec stuff  	mfixName, @@ -221,6 +222,12 @@ basicKnownKeyNames  	-- dotnet interop  	, objectTyConName, marshalObjectName, unmarshalObjectName  	, marshalStringName, unmarshalStringName, checkDotnetResName + +        -- Monad comprehensions +        , guardMName +        , liftMName +        , groupMName +        , mzipName      ]  genericTyConNames :: [Name] @@ -262,8 +269,9 @@ gHC_PRIM, gHC_TYPES, gHC_UNIT, gHC_ORDERING, gHC_GENERICS,      gHC_PACK, gHC_CONC, gHC_IO, gHC_IO_Exception,      gHC_ST, gHC_ARR, gHC_STABLE, gHC_ADDR, gHC_PTR, gHC_ERR, gHC_REAL,      gHC_FLOAT, gHC_TOP_HANDLER, sYSTEM_IO, dYNAMIC, tYPEABLE, gENERICS, -    dOTNET, rEAD_PREC, lEX, gHC_INT, gHC_WORD, mONAD, mONAD_FIX, aRROW, cONTROL_APPLICATIVE, -    gHC_DESUGAR, rANDOM, gHC_EXTS, cONTROL_EXCEPTION_BASE :: Module +    dOTNET, rEAD_PREC, lEX, gHC_INT, gHC_WORD, mONAD, mONAD_FIX, mONAD_GROUP, mONAD_ZIP, +    aRROW, cONTROL_APPLICATIVE, gHC_DESUGAR, rANDOM, gHC_EXTS, +    cONTROL_EXCEPTION_BASE :: Module  gHC_PRIM	= mkPrimModule (fsLit "GHC.Prim")   -- Primitive types and values  gHC_TYPES       = mkPrimModule (fsLit "GHC.Types") @@ -311,6 +319,8 @@ gHC_INT		= mkBaseModule (fsLit "GHC.Int")  gHC_WORD	= mkBaseModule (fsLit "GHC.Word")  mONAD		= mkBaseModule (fsLit "Control.Monad")  mONAD_FIX	= mkBaseModule (fsLit "Control.Monad.Fix") +mONAD_GROUP     = mkBaseModule (fsLit "Control.Monad.Group") +mONAD_ZIP       = mkBaseModule (fsLit "Control.Monad.Zip")  aRROW		= mkBaseModule (fsLit "Control.Arrow")  cONTROL_APPLICATIVE = mkBaseModule (fsLit "Control.Applicative")  gHC_DESUGAR = mkBaseModule (fsLit "GHC.Desugar") @@ -597,12 +607,13 @@ inlineIdName :: Name  inlineIdName	 	= varQual gHC_MAGIC (fsLit "inline") inlineIdKey  -- Base classes (Eq, Ord, Functor) -eqClassName, eqName, ordClassName, geName, functorClassName :: Name +fmapName, eqClassName, eqName, ordClassName, geName, functorClassName :: Name  eqClassName	  = clsQual  gHC_CLASSES (fsLit "Eq")      eqClassKey  eqName		  = methName gHC_CLASSES (fsLit "==")      eqClassOpKey  ordClassName	  = clsQual  gHC_CLASSES (fsLit "Ord")     ordClassKey  geName		  = methName gHC_CLASSES (fsLit ">=")      geClassOpKey  functorClassName  = clsQual  gHC_BASE (fsLit "Functor") functorClassKey +fmapName          = methName gHC_BASE (fsLit "fmap")    fmapClassOpKey  -- Class Monad  monadClassName, thenMName, bindMName, returnMName, failMName :: Name @@ -834,6 +845,14 @@ appAName	   = varQual aRROW (fsLit "app")	  appAIdKey  choiceAName	   = varQual aRROW (fsLit "|||")	  choiceAIdKey  loopAName	   = varQual aRROW (fsLit "loop")  loopAIdKey +-- Monad comprehensions +guardMName, liftMName, groupMName, mzipName :: Name +guardMName         = varQual mONAD (fsLit "guard") guardMIdKey +liftMName          = varQual mONAD (fsLit "liftM") liftMIdKey +groupMName         = varQual mONAD_GROUP (fsLit "mgroupWith") groupMIdKey +mzipName           = varQual mONAD_ZIP (fsLit "mzip") mzipIdKey + +  -- Annotation type checking  toAnnotationWrapperName :: Name  toAnnotationWrapperName = varQual gHC_DESUGAR (fsLit "toAnnotationWrapper") toAnnotationWrapperIdKey @@ -1283,7 +1302,8 @@ unboundKey		      = mkPreludeMiscIdUnique 101  fromIntegerClassOpKey, minusClassOpKey, fromRationalClassOpKey,      enumFromClassOpKey, enumFromThenClassOpKey, enumFromToClassOpKey,      enumFromThenToClassOpKey, eqClassOpKey, geClassOpKey, negateClassOpKey, -    failMClassOpKey, bindMClassOpKey, thenMClassOpKey, returnMClassOpKey +    failMClassOpKey, bindMClassOpKey, thenMClassOpKey, returnMClassOpKey, +    fmapClassOpKey      :: Unique  fromIntegerClassOpKey	      = mkPreludeMiscIdUnique 102  minusClassOpKey		      = mkPreludeMiscIdUnique 103 @@ -1298,6 +1318,7 @@ negateClassOpKey	      = mkPreludeMiscIdUnique 111  failMClassOpKey		      = mkPreludeMiscIdUnique 112  bindMClassOpKey		      = mkPreludeMiscIdUnique 113 -- (>>=)  thenMClassOpKey		      = mkPreludeMiscIdUnique 114 -- (>>) +fmapClassOpKey                = mkPreludeMiscIdUnique 115  returnMClassOpKey	      = mkPreludeMiscIdUnique 117  -- Recursive do notation @@ -1328,6 +1349,14 @@ realToFracIdKey      = mkPreludeMiscIdUnique 128  toIntegerClassOpKey  = mkPreludeMiscIdUnique 129  toRationalClassOpKey = mkPreludeMiscIdUnique 130 +-- Monad comprehensions +guardMIdKey, liftMIdKey, groupMIdKey, mzipIdKey :: Unique +guardMIdKey     = mkPreludeMiscIdUnique 131 +liftMIdKey      = mkPreludeMiscIdUnique 132 +groupMIdKey     = mkPreludeMiscIdUnique 133 +mzipIdKey       = mkPreludeMiscIdUnique 134 + +  ---------------- Template Haskell -------------------  --	USES IdUniques 200-499  ----------------------------------------------------- diff --git a/compiler/rename/RnBinds.lhs b/compiler/rename/RnBinds.lhs index 286e3f2815..63db219a11 100644 --- a/compiler/rename/RnBinds.lhs +++ b/compiler/rename/RnBinds.lhs @@ -789,9 +789,9 @@ rnGRHS' ctxt (GRHS guards rhs)  	-- Standard Haskell 1.4 guards are just a single boolean  	-- expression, rather than a list of qualifiers as in the  	-- Glasgow extension -    is_standard_guard []                     = True -    is_standard_guard [L _ (ExprStmt _ _ _)] = True -    is_standard_guard _                      = False +    is_standard_guard []                       = True +    is_standard_guard [L _ (ExprStmt _ _ _ _)] = True +    is_standard_guard _                        = False  \end{code}  %************************************************************************ diff --git a/compiler/rename/RnExpr.lhs b/compiler/rename/RnExpr.lhs index d11249aea9..46eef670f2 100644 --- a/compiler/rename/RnExpr.lhs +++ b/compiler/rename/RnExpr.lhs @@ -40,7 +40,7 @@ import RdrName  import LoadIface	( loadInterfaceForName )  import UniqSet  import Data.List -import Util		( isSingleton ) +import Util		( isSingleton, snocView )  import ListSetOps	( removeDups )  import Outputable  import SrcLoc @@ -224,10 +224,9 @@ rnExpr (HsLet binds expr)      rnLExpr expr			 `thenM` \ (expr',fvExpr) ->      return (HsLet binds' expr', fvExpr) -rnExpr (HsDo do_or_lc stmts body _) -  = do  { ((stmts', body'), fvs) <- rnStmts do_or_lc stmts $ \ _ -> -				    rnLExpr body -	; return (HsDo do_or_lc stmts' body' placeHolderType, fvs) } +rnExpr (HsDo do_or_lc stmts _) +  = do 	{ ((stmts', _), fvs) <- rnStmts do_or_lc stmts (\ _ -> return ((), emptyFVs)) +	; return ( HsDo do_or_lc stmts' placeHolderType, fvs ) }  rnExpr (ExplicitList _ exps)    = rnExprs exps		 	`thenM` \ (exps', fvs) -> @@ -441,9 +440,9 @@ convertOpFormsCmd (HsIf f exp c1 c2)  convertOpFormsCmd (HsLet binds cmd)    = HsLet binds (convertOpFormsLCmd cmd) -convertOpFormsCmd (HsDo ctxt stmts body ty) -  = HsDo ctxt (map (fmap convertOpFormsStmt) stmts) -	      (convertOpFormsLCmd body) ty +convertOpFormsCmd (HsDo DoExpr stmts ty) +  = HsDo ArrowExpr (map (fmap convertOpFormsStmt) stmts) ty +    -- Mark the HsDo as begin the body of an arrow command  -- Anything else is unchanged.  This includes HsArrForm (already done),  -- things with no sub-commands, and illegal commands (which will be @@ -453,8 +452,8 @@ convertOpFormsCmd c = c  convertOpFormsStmt :: StmtLR id id -> StmtLR id id  convertOpFormsStmt (BindStmt pat cmd _ _)    = BindStmt pat (convertOpFormsLCmd cmd) noSyntaxExpr noSyntaxExpr -convertOpFormsStmt (ExprStmt cmd _ _) -  = ExprStmt (convertOpFormsLCmd cmd) noSyntaxExpr placeHolderType +convertOpFormsStmt (ExprStmt cmd _ _ _) +  = ExprStmt (convertOpFormsLCmd cmd) noSyntaxExpr noSyntaxExpr placeHolderType  convertOpFormsStmt stmt@(RecStmt { recS_stmts = stmts })    = stmt { recS_stmts = map (fmap convertOpFormsStmt) stmts }  convertOpFormsStmt stmt = stmt @@ -495,14 +494,10 @@ methodNamesCmd (HsPar c) = methodNamesLCmd c  methodNamesCmd (HsIf _ _ c1 c2)    = methodNamesLCmd c1 `plusFV` methodNamesLCmd c2 `addOneFV` choiceAName -methodNamesCmd (HsLet _ c) = methodNamesLCmd c - -methodNamesCmd (HsDo _ stmts body _)  -  = methodNamesStmts stmts `plusFV` methodNamesLCmd body - -methodNamesCmd (HsApp c _) = methodNamesLCmd c - -methodNamesCmd (HsLam match) = methodNamesMatch match +methodNamesCmd (HsLet _ c)      = methodNamesLCmd c +methodNamesCmd (HsDo _ stmts _) = methodNamesStmts stmts  +methodNamesCmd (HsApp c _)      = methodNamesLCmd c +methodNamesCmd (HsLam match)    = methodNamesMatch match  methodNamesCmd (HsCase _ matches)    = methodNamesMatch matches `addOneFV` choiceAName @@ -538,14 +533,14 @@ methodNamesLStmt :: Located (StmtLR Name Name) -> FreeVars  methodNamesLStmt = methodNamesStmt . unLoc  methodNamesStmt :: StmtLR Name Name -> FreeVars -methodNamesStmt (ExprStmt cmd _ _)               = methodNamesLCmd cmd +methodNamesStmt (LastStmt cmd _)                 = methodNamesLCmd cmd +methodNamesStmt (ExprStmt cmd _ _ _)             = methodNamesLCmd cmd  methodNamesStmt (BindStmt _ cmd _ _)             = methodNamesLCmd cmd  methodNamesStmt (RecStmt { recS_stmts = stmts }) = methodNamesStmts stmts `addOneFV` loopAName  methodNamesStmt (LetStmt _)                      = emptyFVs -methodNamesStmt (ParStmt _)                      = emptyFVs -methodNamesStmt (TransformStmt {})               = emptyFVs -methodNamesStmt (GroupStmt {})                   = emptyFVs -   -- ParStmt, TransformStmt and GroupStmt can't occur in commands, but it's not convenient to error  +methodNamesStmt (ParStmt _ _ _ _)                = emptyFVs +methodNamesStmt (TransStmt {})                   = emptyFVs +   -- ParStmt and TransStmt can't occur in commands, but it's not convenient to error      -- here so we just do what's convenient  \end{code} @@ -588,14 +583,16 @@ rnArithSeq (FromThenTo expr1 expr2 expr3)  \begin{code}  rnBracket :: HsBracket RdrName -> RnM (HsBracket Name, FreeVars) -rnBracket (VarBr n) = do { name <- lookupOccRn n -			 ; this_mod <- getModule -			 ; unless (nameIsLocalOrFrom this_mod name) $	-- Reason: deprecation checking asumes the -			   do { _ <- loadInterfaceForName msg name	-- home interface is loaded, and this is the -			      ; return () }				-- only way that is going to happen -			 ; return (VarBr name, unitFV name) } -		    where -		      msg = ptext (sLit "Need interface for Template Haskell quoted Name") +rnBracket (VarBr n)  +  = do { name <- lookupOccRn n +       ; this_mod <- getModule +       ; unless (nameIsLocalOrFrom this_mod name) $  -- Reason: deprecation checking assumes +         do { _ <- loadInterfaceForName msg name     -- the home interface is loaded, and +            ; return () }			     -- this is the only way that is going +	      	     				     -- to happen +       ; return (VarBr name, unitFV name) } +  where +    msg = ptext (sLit "Need interface for Template Haskell quoted Name")  rnBracket (ExpBr e) = do { (e', fvs) <- rnLExpr e  			 ; return (ExpBr e', fvs) } @@ -625,7 +622,8 @@ rnBracket (DecBrL decls)  			      rnSrcDecls group        	      -- Discard the tcg_env; it contains only extra info about fixity -        ; traceRn (text "rnBracket dec" <+> (ppr (tcg_dus tcg_env) $$ ppr (duUses (tcg_dus tcg_env)))) +        ; traceRn (text "rnBracket dec" <+> (ppr (tcg_dus tcg_env) $$  +                   ppr (duUses (tcg_dus tcg_env))))  	; return (DecBrG group', duUses (tcg_dus tcg_env)) }  rnBracket (DecBrG _) = panic "rnBracket: unexpected DecBrG" @@ -639,44 +637,72 @@ rnBracket (DecBrG _) = panic "rnBracket: unexpected DecBrG"  \begin{code}  rnStmts :: HsStmtContext Name -> [LStmt RdrName] -	      -> ([Name] -> RnM (thing, FreeVars)) -	      -> RnM (([LStmt Name], thing), FreeVars)	 +	-> ([Name] -> RnM (thing, FreeVars)) +	-> RnM (([LStmt Name], thing), FreeVars)	  -- Variables bound by the Stmts, and mentioned in thing_inside,  -- do not appear in the result FreeVars --- --- Renaming a single RecStmt can give a sequence of smaller Stmts -rnStmts _ [] thing_inside -  = do { (res, fvs) <- thing_inside [] -       ; return (([], res), fvs) } +rnStmts ctxt [] thing_inside +  = do { checkEmptyStmts ctxt +       ; (thing, fvs) <- thing_inside [] +       ; return (([], thing), fvs) } + +rnStmts MDoExpr stmts thing_inside    -- Deal with mdo +  = -- Behave like do { rec { ...all but last... }; last } +    do { ((stmts1, (stmts2, thing)), fvs)  +    	   <- rnStmt MDoExpr (noLoc $ mkRecStmt all_but_last) $ \ _ -> +    	      do { last_stmt' <- checkLastStmt MDoExpr last_stmt +    	         ; rnStmt MDoExpr last_stmt' thing_inside } +	; return (((stmts1 ++ stmts2), thing), fvs) } +  where +    Just (all_but_last, last_stmt) = snocView stmts + +rnStmts ctxt (lstmt@(L loc _) : lstmts) thing_inside +  | null lstmts +  = setSrcSpan loc $ +    do { lstmt' <- checkLastStmt ctxt lstmt +       ; rnStmt ctxt lstmt' thing_inside } -rnStmts ctxt (stmt@(L loc _) : stmts) thing_inside +  | otherwise    = do { ((stmts1, (stmts2, thing)), fvs)  -            <- setSrcSpan loc           $ -               rnStmt ctxt stmt         $ \ bndrs1 -> -               rnStmts ctxt stmts $ \ bndrs2 -> -               thing_inside (bndrs1 ++ bndrs2) +            <- setSrcSpan loc                         $ +               do { checkStmt ctxt lstmt +                  ; rnStmt ctxt lstmt    $ \ bndrs1 -> +                    rnStmts ctxt lstmts  $ \ bndrs2 -> +                    thing_inside (bndrs1 ++ bndrs2) }  	; return (((stmts1 ++ stmts2), thing), fvs) } - -rnStmt :: HsStmtContext Name -> LStmt RdrName +---------------------- +rnStmt :: HsStmtContext Name  +       -> LStmt RdrName         -> ([Name] -> RnM (thing, FreeVars))         -> RnM (([LStmt Name], thing), FreeVars)  -- Variables bound by the Stmt, and mentioned in thing_inside,  -- do not appear in the result FreeVars -rnStmt _ (L loc (ExprStmt expr _ _)) thing_inside +rnStmt ctxt (L loc (LastStmt expr _)) thing_inside    = do	{ (expr', fv_expr) <- rnLExpr expr -	; (then_op, fvs1)  <- lookupSyntaxName thenMName -	; (thing, fvs2)    <- thing_inside [] -	; return (([L loc (ExprStmt expr' then_op placeHolderType)], thing), -		  fv_expr `plusFV` fvs1 `plusFV` fvs2) } +	; (ret_op, fvs1)   <- lookupStmtName ctxt returnMName +	; (thing,  fvs3)   <- thing_inside [] +	; return (([L loc (LastStmt expr' ret_op)], thing), +		  fv_expr `plusFV` fvs1 `plusFV` fvs3) } + +rnStmt ctxt (L loc (ExprStmt expr _ _ _)) thing_inside +  = do	{ (expr', fv_expr) <- rnLExpr expr +	; (then_op, fvs1)  <- lookupStmtName ctxt thenMName +	; (guard_op, fvs2) <- if isListCompExpr ctxt +                              then lookupStmtName ctxt guardMName +			      else return (noSyntaxExpr, emptyFVs) +			      -- Only list/parr/monad comprehensions use 'guard' +	; (thing, fvs3)    <- thing_inside [] +	; return (([L loc (ExprStmt expr' then_op guard_op placeHolderType)], thing), +		  fv_expr `plusFV` fvs1 `plusFV` fvs2 `plusFV` fvs3) }  rnStmt ctxt (L loc (BindStmt pat expr _ _)) thing_inside    = do	{ (expr', fv_expr) <- rnLExpr expr  		-- The binders do not scope over the expression -	; (bind_op, fvs1) <- lookupSyntaxName bindMName -	; (fail_op, fvs2) <- lookupSyntaxName failMName +	; (bind_op, fvs1) <- lookupStmtName ctxt bindMName +	; (fail_op, fvs2) <- lookupStmtName ctxt failMName  	; rnPat (StmtCtxt ctxt) pat $ \ pat' -> do  	{ (thing, fvs3) <- thing_inside (collectPatBinders pat')  	; return (([L loc (BindStmt pat' expr' bind_op fail_op)], thing), @@ -684,15 +710,13 @@ rnStmt ctxt (L loc (BindStmt pat expr _ _)) thing_inside         -- fv_expr shouldn't really be filtered by the rnPatsAndThen  	-- but it does not matter because the names are unique -rnStmt ctxt (L loc (LetStmt binds)) thing_inside  -  = do	{ checkLetStmt ctxt binds -	; rnLocalBindsAndThen binds $ \binds' -> do +rnStmt _ (L loc (LetStmt binds)) thing_inside  +  = do	{ rnLocalBindsAndThen binds $ \binds' -> do  	{ (thing, fvs) <- thing_inside (collectLocalBinders binds')          ; return (([L loc (LetStmt binds')], thing), fvs) }  }  rnStmt ctxt (L _ (RecStmt { recS_stmts = rec_stmts })) thing_inside -  = do	{ checkRecStmt ctxt - +  = do	{   	-- Step1: Bring all the binders of the mdo into scope  	-- (Remember that this also removes the binders from the  	-- finally-returned free-vars.) @@ -707,9 +731,9 @@ rnStmt ctxt (L _ (RecStmt { recS_stmts = rec_stmts })) thing_inside  	{ let bndrs = nameSetToList $ foldr (unionNameSets . (\(ds,_,_,_) -> ds))                                               emptyNameSet segs          ; (thing, fvs_later) <- thing_inside bndrs -	; (return_op, fvs1)  <- lookupSyntaxName returnMName -	; (mfix_op,   fvs2)  <- lookupSyntaxName mfixName -	; (bind_op,   fvs3)  <- lookupSyntaxName bindMName +	; (return_op, fvs1)  <- lookupStmtName ctxt returnMName +	; (mfix_op,   fvs2)  <- lookupStmtName ctxt mfixName +	; (bind_op,   fvs3)  <- lookupStmtName ctxt bindMName  	; let  		-- Step 2: Fill in the fwd refs.  		-- 	   The segments are all singletons, but their fwd-ref @@ -734,57 +758,51 @@ rnStmt ctxt (L _ (RecStmt { recS_stmts = rec_stmts })) thing_inside  	; return ((rec_stmts', thing), fvs `plusFV` fvs1 `plusFV` fvs2 `plusFV` fvs3) } } -rnStmt ctxt (L loc (ParStmt segs)) thing_inside -  = do	{ checkParStmt ctxt -	; ((segs', thing), fvs) <- rnParallelStmts (ParStmtCtxt ctxt) segs thing_inside -	; return (([L loc (ParStmt segs')], thing), fvs) } - -rnStmt ctxt (L loc (TransformStmt stmts _ using by)) thing_inside -  = do { checkTransformStmt ctxt -     -       ; (using', fvs1) <- rnLExpr using - -       ; ((stmts', (by', used_bndrs, thing)), fvs2) -             <- rnStmts (TransformStmtCtxt ctxt) stmts $ \ bndrs -> -                do { (by', fvs_by) <- case by of -                                        Nothing -> return (Nothing, emptyFVs) -                                        Just e  -> do { (e', fvs) <- rnLExpr e; return (Just e', fvs) } -                   ; (thing, fvs_thing) <- thing_inside bndrs -                   ; let fvs        = fvs_by `plusFV` fvs_thing -                         used_bndrs = filter (`elemNameSet` fvs) bndrs -                         -- The paper (Fig 5) has a bug here; we must treat any free varaible of -                         -- the "thing inside", **or of the by-expression**, as used -                   ; return ((by', used_bndrs, thing), fvs) } - -       ; return (([L loc (TransformStmt stmts' used_bndrs using' by')], thing),  -                 fvs1 `plusFV` fvs2) } -         -rnStmt ctxt (L loc (GroupStmt stmts _ by using)) thing_inside -  = do { checkTransformStmt ctxt -     -         -- Rename the 'using' expression in the context before the transform is begun -       ; (using', fvs1) <- case using of -                             Left e  -> do { (e', fvs) <- rnLExpr e; return (Left e', fvs) } -			     Right _ -> do { (e', fvs) <- lookupSyntaxName groupWithName -                                           ; return (Right e', fvs) } +rnStmt ctxt (L loc (ParStmt segs _ _ _)) thing_inside +  = do	{ (mzip_op, fvs1)   <- lookupStmtName ctxt mzipName +        ; (bind_op, fvs2)   <- lookupStmtName ctxt bindMName +        ; (return_op, fvs3) <- lookupStmtName ctxt returnMName +	; ((segs', thing), fvs4) <- rnParallelStmts (ParStmtCtxt ctxt) segs thing_inside +	; return ( ([L loc (ParStmt segs' mzip_op bind_op return_op)], thing) +                 , fvs1 `plusFV` fvs2 `plusFV` fvs3 `plusFV` fvs4) } + +rnStmt ctxt (L loc (TransStmt { trS_stmts = stmts, trS_by = by, trS_form = form +                              , trS_using = using })) thing_inside +  = do { -- Rename the 'using' expression in the context before the transform is begun +         (using', fvs1) <- case form of +                             GroupFormB -> do { (e,fvs) <- lookupStmtName ctxt groupMName +                                              ; return (noLoc e, fvs) } +			     _          -> rnLExpr using           -- Rename the stmts and the 'by' expression  	 -- Keep track of the variables mentioned in the 'by' expression         ; ((stmts', (by', used_bndrs, thing)), fvs2)  -             <- rnStmts (TransformStmtCtxt ctxt) stmts $ \ bndrs -> +             <- rnStmts (TransStmtCtxt ctxt) stmts $ \ bndrs ->                  do { (by',   fvs_by) <- mapMaybeFvRn rnLExpr by                     ; (thing, fvs_thing) <- thing_inside bndrs                     ; let fvs = fvs_by `plusFV` fvs_thing                           used_bndrs = filter (`elemNameSet` fvs) bndrs +                         -- The paper (Fig 5) has a bug here; we must treat any free varaible +                         -- of the "thing inside", **or of the by-expression**, as used                     ; return ((by', used_bndrs, thing), fvs) } -       ; let all_fvs  = fvs1 `plusFV` fvs2  +       -- Lookup `return`, `(>>=)` and `liftM` for monad comprehensions +       ; (return_op, fvs3) <- lookupStmtName ctxt returnMName +       ; (bind_op,   fvs4) <- lookupStmtName ctxt bindMName +       ; (fmap_op,   fvs5) <- case form of +                                ThenForm -> return (noSyntaxExpr, emptyFVs) +                                _        -> lookupStmtName ctxt fmapName + +       ; let all_fvs  = fvs1 `plusFV` fvs2 `plusFV` fvs3  +                             `plusFV` fvs4 `plusFV` fvs5               bndr_map = used_bndrs `zip` used_bndrs -	     -- See Note [GroupStmt binder map] in HsExpr +	     -- See Note [TransStmt binder map] in HsExpr         ; traceRn (text "rnStmt: implicitly rebound these used binders:" <+> ppr bndr_map) -       ; return (([L loc (GroupStmt stmts' bndr_map by' using')], thing), all_fvs) } - +       ; return (([L loc (TransStmt { trS_stmts = stmts', trS_bndrs = bndr_map +                                    , trS_by = by', trS_using = using', trS_form = form +                                    , trS_ret = return_op, trS_bind = bind_op +                                    , trS_fmap = fmap_op })], thing), all_fvs) }  type ParSeg id = ([LStmt id], [id])	   -- The Names are bound by the Stmts @@ -820,6 +838,27 @@ rnParallelStmts ctxt segs thing_inside      cmpByOcc n1 n2 = nameOccName n1 `compare` nameOccName n2      dupErr vs = addErr (ptext (sLit "Duplicate binding in parallel list comprehension for:")                      <+> quotes (ppr (head vs))) + +lookupStmtName :: HsStmtContext Name -> Name -> RnM (HsExpr Name, FreeVars) +-- Like lookupSyntaxName, but ListComp/PArrComp are never rebindable +-- Neither is ArrowExpr, which has its own desugarer in DsArrows +lookupStmtName ctxt n  +  = case ctxt of +      ListComp        -> not_rebindable +      PArrComp        -> not_rebindable +      ArrowExpr       -> not_rebindable +      PatGuard {}     -> not_rebindable + +      DoExpr          -> rebindable +      MDoExpr         -> rebindable +      MonadComp       -> rebindable +      GhciStmt        -> rebindable   -- I suppose? + +      ParStmtCtxt   c -> lookupStmtName c n	-- Look inside to +      TransStmtCtxt c -> lookupStmtName c n	-- the parent context +  where +    rebindable     = lookupSyntaxName n +    not_rebindable = return (HsVar n, emptyFVs)  \end{code}  Note [Renaming parallel Stmts] @@ -901,9 +940,11 @@ rn_rec_stmt_lhs :: MiniFixityEnv                     -- so we don't bother to compute it accurately in the other cases                  -> RnM [(LStmtLR Name RdrName, FreeVars)] -rn_rec_stmt_lhs _ (L loc (ExprStmt expr a b)) = return [(L loc (ExprStmt expr a b),  -                                                       -- this is actually correct -                                                       emptyFVs)] +rn_rec_stmt_lhs _ (L loc (ExprStmt expr a b c))  +  = return [(L loc (ExprStmt expr a b c), emptyFVs)] + +rn_rec_stmt_lhs _ (L loc (LastStmt expr a))  +  = return [(L loc (LastStmt expr a), emptyFVs)]  rn_rec_stmt_lhs fix_env (L loc (BindStmt pat expr a b))     = do  @@ -926,13 +967,10 @@ rn_rec_stmt_lhs fix_env (L loc (LetStmt (HsValBinds binds)))  rn_rec_stmt_lhs fix_env (L _ (RecStmt { recS_stmts = stmts }))	-- Flatten Rec inside Rec      = rn_rec_stmts_lhs fix_env stmts -rn_rec_stmt_lhs _ stmt@(L _ (ParStmt _))	-- Syntactically illegal in mdo -  = pprPanic "rn_rec_stmt" (ppr stmt) -   -rn_rec_stmt_lhs _ stmt@(L _ (TransformStmt {}))	-- Syntactically illegal in mdo +rn_rec_stmt_lhs _ stmt@(L _ (ParStmt _ _ _ _))	-- Syntactically illegal in mdo    = pprPanic "rn_rec_stmt" (ppr stmt) -rn_rec_stmt_lhs _ stmt@(L _ (GroupStmt {}))	-- Syntactically illegal in mdo +rn_rec_stmt_lhs _ stmt@(L _ (TransStmt {}))	-- Syntactically illegal in mdo    = pprPanic "rn_rec_stmt" (ppr stmt)  rn_rec_stmt_lhs _ (L _ (LetStmt EmptyLocalBinds)) @@ -957,11 +995,17 @@ rn_rec_stmt :: [Name] -> LStmtLR Name RdrName -> FreeVars -> RnM [Segment (LStmt  	-- Rename a Stmt that is inside a RecStmt (or mdo)  	-- Assumes all binders are already in scope  	-- Turns each stmt into a singleton Stmt -rn_rec_stmt _ (L loc (ExprStmt expr _ _)) _ +rn_rec_stmt _ (L loc (LastStmt expr _)) _ +  = do	{ (expr', fv_expr) <- rnLExpr expr +	; (ret_op, fvs1)   <- lookupSyntaxName returnMName +	; return [(emptyNameSet, fv_expr `plusFV` fvs1, emptyNameSet, +                   L loc (LastStmt expr' ret_op))] } + +rn_rec_stmt _ (L loc (ExprStmt expr _ _ _)) _    = rnLExpr expr `thenM` \ (expr', fvs) ->      lookupSyntaxName thenMName	`thenM` \ (then_op, fvs1) ->      return [(emptyNameSet, fvs `plusFV` fvs1, emptyNameSet, -	      L loc (ExprStmt expr' then_op placeHolderType))] +	      L loc (ExprStmt expr' then_op noSyntaxExpr placeHolderType))]  rn_rec_stmt _ (L loc (BindStmt pat' expr _ _)) fv_pat    = rnLExpr expr		`thenM` \ (expr', fv_expr) -> @@ -991,11 +1035,8 @@ rn_rec_stmt _ stmt@(L _ (RecStmt {})) _  rn_rec_stmt _ stmt@(L _ (ParStmt {})) _	-- Syntactically illegal in mdo    = pprPanic "rn_rec_stmt: ParStmt" (ppr stmt) -rn_rec_stmt _ stmt@(L _ (TransformStmt {})) _	-- Syntactically illegal in mdo -  = pprPanic "rn_rec_stmt: TransformStmt" (ppr stmt) - -rn_rec_stmt _ stmt@(L _ (GroupStmt {})) _	-- Syntactically illegal in mdo -  = pprPanic "rn_rec_stmt: GroupStmt" (ppr stmt) +rn_rec_stmt _ stmt@(L _ (TransStmt {})) _	-- Syntactically illegal in mdo +  = pprPanic "rn_rec_stmt: TransStmt" (ppr stmt)  rn_rec_stmt _ (L _ (LetStmt EmptyLocalBinds)) _    = panic "rn_rec_stmt: LetStmt EmptyLocalBinds" @@ -1141,44 +1182,151 @@ program.  %************************************************************************  \begin{code} +checkEmptyStmts :: HsStmtContext Name -> RnM () +-- We've seen an empty sequence of Stmts... is that ok? +checkEmptyStmts ctxt  +  = unless (okEmpty ctxt) (addErr (emptyErr ctxt)) -----------------------  --- Checking when a particular Stmt is ok -checkLetStmt :: HsStmtContext Name -> HsLocalBinds RdrName -> RnM () -checkLetStmt (ParStmtCtxt _) (HsIPBinds binds) = addErr (badIpBinds (ptext (sLit "a parallel list comprehension:")) binds) -checkLetStmt _ctxt 	     _binds	       = return () -  	-- We do not allow implicit-parameter bindings in a parallel -	-- list comprehension.  I'm not sure what it might mean. +okEmpty :: HsStmtContext a -> Bool +okEmpty (PatGuard {}) = True +okEmpty _             = False ---------- -checkRecStmt :: HsStmtContext Name -> RnM () -checkRecStmt MDoExpr = return ()      -- Recursive stmt ok in 'mdo' -checkRecStmt DoExpr  = return ()      -- and in 'do' -checkRecStmt ctxt    = addErr msg -  where -    msg = ptext (sLit "Illegal 'rec' stmt in") <+> pprStmtContext ctxt +emptyErr :: HsStmtContext Name -> SDoc +emptyErr (ParStmtCtxt {})   = ptext (sLit "Empty statement group in parallel comprehension") +emptyErr (TransStmtCtxt {}) = ptext (sLit "Empty statement group preceding 'group' or 'then'") +emptyErr ctxt               = ptext (sLit "Empty") <+> pprStmtContext ctxt ---------- -checkParStmt :: HsStmtContext Name -> RnM () -checkParStmt _ -  = do	{ parallel_list_comp <- xoptM Opt_ParallelListComp -	; checkErr parallel_list_comp msg } +----------------------  +checkLastStmt :: HsStmtContext Name +              -> LStmt RdrName  +              -> RnM (LStmt RdrName) +checkLastStmt ctxt lstmt@(L loc stmt) +  = case ctxt of  +      ListComp  -> check_comp +      MonadComp -> check_comp +      PArrComp  -> check_comp +      ArrowExpr	-> check_do +      DoExpr	-> check_do +      MDoExpr   -> check_do +      _         -> check_other    where -    msg = ptext (sLit "Illegal parallel list comprehension: use -XParallelListComp") +    check_do	-- Expect ExprStmt, and change it to LastStmt +      = case stmt of  +          ExprStmt e _ _ _ -> return (L loc (mkLastStmt e)) +          LastStmt {}      -> return lstmt   -- "Deriving" clauses may generate a +	  	   	      	     	     -- LastStmt directly (unlike the parser) +	  _                -> do { addErr (hang last_error 2 (ppr stmt)); return lstmt } +    last_error = (ptext (sLit "The last statement in") <+> pprAStmtContext ctxt +                  <+> ptext (sLit "must be an expression")) + +    check_comp	-- Expect LastStmt; this should be enforced by the parser! +      = case stmt of  +          LastStmt {} -> return lstmt +          _           -> pprPanic "checkLastStmt" (ppr lstmt) + +    check_other	-- Behave just as if this wasn't the last stmt +      = do { checkStmt ctxt lstmt; return lstmt } ---------- -checkTransformStmt :: HsStmtContext Name -> RnM () -checkTransformStmt ListComp  -- Ensure we are really within a list comprehension because otherwise the -			     -- desugarer will break when we come to operate on a parallel array -  = do	{ transform_list_comp <- xoptM Opt_TransformListComp -	; checkErr transform_list_comp msg } -  where -    msg = ptext (sLit "Illegal transform or grouping list comprehension: use -XTransformListComp") -checkTransformStmt (ParStmtCtxt       ctxt) = checkTransformStmt ctxt	-- Ok to nest inside a parallel comprehension -checkTransformStmt (TransformStmtCtxt ctxt) = checkTransformStmt ctxt	-- Ok to nest inside a parallel comprehension -checkTransformStmt ctxt = addErr msg +-- Checking when a particular Stmt is ok +checkStmt :: HsStmtContext Name +          -> LStmt RdrName  +          -> RnM () +checkStmt ctxt (L _ stmt) +  = do { dflags <- getDOpts +       ; case okStmt dflags ctxt stmt of  +           Nothing    -> return () +           Just extra -> addErr (msg $$ extra) }    where -    msg = ptext (sLit "Illegal transform or grouping in") <+> pprStmtContext ctxt +   msg = sep [ ptext (sLit "Unexpected") <+> pprStmtCat stmt <+> ptext (sLit "statement") +             , ptext (sLit "in") <+> pprAStmtContext ctxt ] + +pprStmtCat :: Stmt a -> SDoc +pprStmtCat (TransStmt {})     = ptext (sLit "transform") +pprStmtCat (LastStmt {})      = ptext (sLit "return expression") +pprStmtCat (ExprStmt {})      = ptext (sLit "exprssion") +pprStmtCat (BindStmt {})      = ptext (sLit "binding") +pprStmtCat (LetStmt {})       = ptext (sLit "let") +pprStmtCat (RecStmt {})       = ptext (sLit "rec") +pprStmtCat (ParStmt {})       = ptext (sLit "parallel") + +------------ +isOK, notOK :: Maybe SDoc +isOK  = Nothing +notOK = Just empty + +okStmt, okDoStmt, okCompStmt, okParStmt, okPArrStmt +   :: DynFlags -> HsStmtContext Name +   -> Stmt RdrName -> Maybe SDoc +-- Return Nothing if OK, (Just extra) if not ok +-- The "extra" is an SDoc that is appended to an generic error message + +okStmt dflags ctxt stmt  +  = case ctxt of +      PatGuard {}      	 -> okPatGuardStmt stmt +      ParStmtCtxt ctxt 	 -> okParStmt  dflags ctxt stmt +      DoExpr           	 -> okDoStmt   dflags ctxt stmt +      MDoExpr          	 -> okDoStmt   dflags ctxt stmt +      ArrowExpr        	 -> okDoStmt   dflags ctxt stmt +      GhciStmt         	 -> okDoStmt   dflags ctxt stmt +      ListComp         	 -> okCompStmt dflags ctxt stmt +      MonadComp        	 -> okCompStmt dflags ctxt stmt +      PArrComp         	 -> okPArrStmt dflags ctxt stmt +      TransStmtCtxt ctxt -> okStmt dflags ctxt stmt + +------------- +okPatGuardStmt :: Stmt RdrName -> Maybe SDoc +okPatGuardStmt stmt +  = case stmt of +      ExprStmt {} -> isOK +      BindStmt {} -> isOK +      LetStmt {}  -> isOK +      _           -> notOK + +------------- +okParStmt dflags ctxt stmt +  = case stmt of +      LetStmt (HsIPBinds {}) -> notOK +      _                      -> okStmt dflags ctxt stmt + +---------------- +okDoStmt dflags ctxt stmt +  = case stmt of +       RecStmt {} +         | Opt_DoRec `xopt` dflags -> isOK +         | ArrowExpr <- ctxt       -> isOK	-- Arrows allows 'rec' +         | otherwise               -> Just (ptext (sLit "Use -XDoRec")) +       BindStmt {} -> isOK +       LetStmt {}  -> isOK +       ExprStmt {} -> isOK +       _           -> notOK + +---------------- +okCompStmt dflags _ stmt +  = case stmt of +       BindStmt {} -> isOK +       LetStmt {}  -> isOK +       ExprStmt {} -> isOK +       ParStmt {}  +         | Opt_ParallelListComp `xopt` dflags -> isOK +         | otherwise -> Just (ptext (sLit "Use -XParallelListComp")) +       TransStmt {}  +         | Opt_TransformListComp `xopt` dflags -> isOK +         | otherwise -> Just (ptext (sLit "Use -XTransformListComp")) +       RecStmt {}  -> notOK +       LastStmt {} -> notOK  -- Should not happen (dealt with by checkLastStmt) + +---------------- +okPArrStmt dflags _ stmt +  = case stmt of +       BindStmt {} -> isOK +       LetStmt {}  -> isOK +       ExprStmt {} -> isOK +       ParStmt {}  +         | Opt_ParallelListComp `xopt` dflags -> isOK +         | otherwise -> Just (ptext (sLit "Use -XParallelListComp")) +       TransStmt {} -> notOK +       RecStmt {}   -> notOK +       LastStmt {}  -> notOK  -- Should not happen (dealt with by checkLastStmt)  ---------  checkTupleSection :: [HsTupArg RdrName] -> RnM () diff --git a/compiler/typecheck/TcArrows.lhs b/compiler/typecheck/TcArrows.lhs index de236e767c..7ce5fc1a57 100644 --- a/compiler/typecheck/TcArrows.lhs +++ b/compiler/typecheck/TcArrows.lhs @@ -7,7 +7,7 @@ Typecheck arrow notation  \begin{code}  module TcArrows ( tcProc ) where -import {-# SOURCE #-}	TcExpr( tcMonoExpr, tcInferRho, tcSyntaxOp ) +import {-# SOURCE #-}	TcExpr( tcMonoExpr, tcInferRho, tcSyntaxOp, tcCheckId )  import HsSyn  import TcMatches @@ -17,7 +17,9 @@ import TcBinds  import TcPat  import TcUnify  import TcRnMonad +import TcEnv  import Coercion +import Id( mkLocalId )  import Inst  import Name  import TysWiredIn @@ -83,20 +85,12 @@ tcCmdTop :: CmdEnv  tcCmdTop env (L loc (HsCmdTop cmd _ _ names)) cmd_stk res_ty    = setSrcSpan loc $ -    do	{ cmd'   <- tcGuardedCmd env cmd cmd_stk res_ty +    do	{ cmd'   <- tcCmd env cmd (cmd_stk, res_ty)  	; names' <- mapM (tcSyntaxName ProcOrigin (cmd_arr env)) names  	; return (L loc $ HsCmdTop cmd' cmd_stk res_ty names') }  ---------------------------------------- -tcGuardedCmd :: CmdEnv -> LHsExpr Name -> CmdStack -	     -> TcTauType -> TcM (LHsExpr TcId) --- A wrapper that deals with the refinement (if any) -tcGuardedCmd env expr stk res_ty -  = do	{ body <- tcCmd env expr (stk, res_ty) -	; return body  -        } -  tcCmd :: CmdEnv -> LHsExpr Name -> (CmdStack, TcTauType) -> TcM (LHsExpr TcId)  	-- The main recursive function  tcCmd env (L loc expr) res_ty @@ -123,7 +117,7 @@ tc_cmd env in_cmd@(HsCase scrut matches) (stk, res_ty)    where      match_ctxt = MC { mc_what = CaseAlt,                        mc_body = mc_body } -    mc_body body res_ty' = tcGuardedCmd env body stk res_ty' +    mc_body body res_ty' = tcCmd env body (stk, res_ty')  tc_cmd env (HsIf mb_fun pred b1 b2) (stack_ty,res_ty)    = do 	{ pred_ty <- newFlexiTyVarTy openTypeKind @@ -206,22 +200,18 @@ tc_cmd env cmd@(HsLam (MatchGroup [L mtch_loc (match@(Match pats _maybe_rhs_sig  	     ; return (GRHSs grhss' binds') }      tc_grhs res_ty (GRHS guards body) -	= do { (guards', rhs') <- tcStmts pg_ctxt tcGuardStmt guards res_ty $ -				  tcGuardedCmd env body stk' +	= do { (guards', rhs') <- tcStmtsAndThen pg_ctxt tcGuardStmt guards res_ty $ +				  \ res_ty -> tcCmd env body (stk', res_ty)  	     ; return (GRHS guards' rhs') }  -------------------------------------------  -- 		Do notation -tc_cmd env cmd@(HsDo do_or_lc stmts body _ty) (cmd_stk, res_ty) +tc_cmd env cmd@(HsDo do_or_lc stmts _) (cmd_stk, res_ty)    = do 	{ checkTc (null cmd_stk) (nonEmptyCmdStkErr cmd) -	; (stmts', body') <- tcStmts do_or_lc (tcMDoStmt tc_rhs) stmts res_ty $ -			     tcGuardedCmd env body [] -	; return (HsDo do_or_lc stmts' body' res_ty) } +	; stmts' <- tcStmts do_or_lc (tcArrDoStmt env) stmts res_ty  +	; return (HsDo do_or_lc stmts' res_ty) }    where -    tc_rhs rhs = do { ty <- newFlexiTyVarTy liftedTypeKind -		    ; rhs' <- tcCmd env rhs ([], ty) -		    ; return (rhs', ty) }  ----------------------------------------------------------------- @@ -307,6 +297,69 @@ tc_cmd _ cmd _  %************************************************************************  %*									* +		Stmts +%*									* +%************************************************************************ + +\begin{code} +-------------------------------- +--	Mdo-notation +-- The distinctive features here are +--	(a) RecStmts, and +--	(b) no rebindable syntax + +tcArrDoStmt :: CmdEnv -> TcStmtChecker +tcArrDoStmt env _ (LastStmt rhs _) res_ty thing_inside +  = do	{ rhs' <- tcCmd env rhs ([], res_ty) +	; thing <- thing_inside (panic "tcArrDoStmt") +	; return (LastStmt rhs' noSyntaxExpr, thing) } + +tcArrDoStmt env _ (ExprStmt rhs _ _ _) res_ty thing_inside +  = do	{ (rhs', elt_ty) <- tc_arr_rhs env rhs +	; thing 	 <- thing_inside res_ty +	; return (ExprStmt rhs' noSyntaxExpr noSyntaxExpr elt_ty, thing) } + +tcArrDoStmt env ctxt (BindStmt pat rhs _ _) res_ty thing_inside +  = do	{ (rhs', pat_ty) <- tc_arr_rhs env rhs +	; (pat', thing)  <- tcPat (StmtCtxt ctxt) pat pat_ty $ +                            thing_inside res_ty +	; return (BindStmt pat' rhs' noSyntaxExpr noSyntaxExpr, thing) } + +tcArrDoStmt env ctxt (RecStmt { recS_stmts = stmts, recS_later_ids = laterNames +                            , recS_rec_ids = recNames }) res_ty thing_inside +  = do	{ rec_tys <- newFlexiTyVarTys (length recNames) liftedTypeKind +	; let rec_ids = zipWith mkLocalId recNames rec_tys +	; tcExtendIdEnv rec_ids $ do +    	{ (stmts', (later_ids, rec_rets)) +		<- tcStmtsAndThen ctxt (tcArrDoStmt env) stmts res_ty	$ \ _res_ty' -> +			-- ToDo: res_ty not really right +		   do { rec_rets <- zipWithM tcCheckId recNames rec_tys +		      ; later_ids <- tcLookupLocalIds laterNames +		      ; return (later_ids, rec_rets) } + +	; thing <- tcExtendIdEnv later_ids (thing_inside res_ty) +		-- NB:	The rec_ids for the recursive things  +		-- 	already scope over this part. This binding may shadow +		--	some of them with polymorphic things with the same Name +		--	(see note [RecStmt] in HsExpr) + +        ; return (emptyRecStmt { recS_stmts = stmts', recS_later_ids = later_ids +                               , recS_rec_ids = rec_ids, recS_rec_rets = rec_rets +                               , recS_ret_ty = res_ty }, thing) +	}} + +tcArrDoStmt _ _ stmt _ _ +  = pprPanic "tcArrDoStmt: unexpected Stmt" (ppr stmt) + +tc_arr_rhs :: CmdEnv -> LHsExpr Name -> TcM (LHsExpr TcId, TcType) +tc_arr_rhs env rhs = do { ty <- newFlexiTyVarTy liftedTypeKind +		        ; rhs' <- tcCmd env rhs ([], ty) +		        ; return (rhs', ty) } +\end{code} + + +%************************************************************************ +%*									*  		Helpers  %*									*  %************************************************************************ diff --git a/compiler/typecheck/TcExpr.lhs b/compiler/typecheck/TcExpr.lhs index 2236740407..ee6a34ac06 100644 --- a/compiler/typecheck/TcExpr.lhs +++ b/compiler/typecheck/TcExpr.lhs @@ -45,6 +45,7 @@ import Type  import Coercion  import Var  import VarSet +import VarEnv  import TysWiredIn  import TysPrim( intPrimTy )  import PrimOp( tagToEnumKey ) @@ -55,6 +56,7 @@ import SrcLoc  import Util  import ListSetOps  import Maybes +import ErrUtils  import Outputable  import FastString  import Control.Monad @@ -415,8 +417,8 @@ tcExpr (HsIf (Just fun) pred b1 b2) res_ty   -- Note [Rebindable syntax for if]         -- and it maintains uniformity with other rebindable syntax         ; return (HsIf (Just fun') pred' b1' b2') } -tcExpr (HsDo do_or_lc stmts body _) res_ty -  = tcDoStmts do_or_lc stmts body res_ty +tcExpr (HsDo do_or_lc stmts _) res_ty +  = tcDoStmts do_or_lc stmts res_ty  tcExpr (HsProc pat cmd) res_ty    = do	{ (pat', cmd', coi) <- tcProc pat cmd res_ty @@ -820,7 +822,7 @@ tcApp fun args res_ty  	-- Typecheck the result, thereby propagating           -- info (if any) from result into the argument types          -- Both actual_res_ty and res_ty are deeply skolemised -        ; co_res <- addErrCtxt (funResCtxt fun) $ +        ; co_res <- addErrCtxtM (funResCtxt fun actual_res_ty res_ty) $                      unifyType actual_res_ty res_ty  	-- Typecheck the arguments @@ -1386,9 +1388,23 @@ funAppCtxt fun arg arg_no  		    quotes (ppr fun) <> text ", namely"])         2 (quotes (ppr arg)) -funResCtxt :: LHsExpr Name -> SDoc -funResCtxt fun -  = ptext (sLit "In the return type of a call of") <+> quotes (ppr fun) +funResCtxt :: LHsExpr Name -> TcType -> TcType  +           -> TidyEnv -> TcM (TidyEnv, Message) +-- When we have a mis-match in the return type of a function +-- try to give a helpful message about too many/few arguments +funResCtxt fun fun_res_ty res_ty env0 +  = do { fun_res' <- zonkTcType fun_res_ty +       ; res'     <- zonkTcType res_ty +       ; let n_fun = length (fst (tcSplitFunTys fun_res')) +             n_res = length (fst (tcSplitFunTys res')) +             what  | n_fun > n_res = ptext (sLit "few") +                   | otherwise     = ptext (sLit "many") +             extra | n_fun == n_res = empty +                   | otherwise = ptext (sLit "Probable cause:") <+> quotes (ppr fun) +                                 <+> ptext (sLit "is applied to too") <+> what  +                                 <+> ptext (sLit "arguments")  +             msg = ptext (sLit "In the return type of a call of") <+> quotes (ppr fun) +       ; return (env0, msg $$ extra) }  badFieldTypes :: [(Name,TcType)] -> SDoc  badFieldTypes prs diff --git a/compiler/typecheck/TcGenDeriv.lhs b/compiler/typecheck/TcGenDeriv.lhs index b76b75cb7f..310f3fd2c4 100644 --- a/compiler/typecheck/TcGenDeriv.lhs +++ b/compiler/typecheck/TcGenDeriv.lhs @@ -778,7 +778,7 @@ gen_Ix_binds loc tycon      single_con_range        = mk_easy_FunBind loc range_RDR   	  [nlTuplePat [con_pat as_needed, con_pat bs_needed] Boxed] $ -	nlHsDo ListComp stmts con_expr +	noLoc (mkHsComp ListComp stmts con_expr)        where  	stmts = zipWith3Equal "single_con_range" mk_qual as_needed bs_needed cs_needed @@ -892,7 +892,7 @@ gen_Read_binds get_fixity loc tycon      read_nullary_cons         = case nullary_cons of      	    []    -> [] -    	    [con] -> [nlHsDo DoExpr (match_con con) (result_expr con [])] +    	    [con] -> [nlHsDo DoExpr (match_con con ++ [noLoc $ mkLastStmt (result_expr con [])])]              _     -> [nlHsApp (nlHsVar choose_RDR)       		   	      (nlList (map mk_pair nullary_cons))]          -- NB For operators the parens around (:=:) are matched by the @@ -964,11 +964,12 @@ gen_Read_binds get_fixity loc tycon      ------------------------------------------------------------------------      --		Helpers      ------------------------------------------------------------------------ -    mk_alt e1 e2       = genOpApp e1 alt_RDR e2					-- e1 +++ e2 -    mk_parser p ss b   = nlHsApps prec_RDR [nlHsIntLit p, nlHsDo DoExpr ss b]	-- prec p (do { ss ; b }) -    bindLex pat	       = noLoc (mkBindStmt pat (nlHsVar lexP_RDR))		-- pat <- lexP -    con_app con as     = nlHsVarApps (getRdrName con) as			-- con as -    result_expr con as = nlHsApp (nlHsVar returnM_RDR) (con_app con as)		-- return (con as) +    mk_alt e1 e2       = genOpApp e1 alt_RDR e2				-- e1 +++ e2 +    mk_parser p ss b   = nlHsApps prec_RDR [nlHsIntLit p	        -- prec p (do { ss ; b }) +                                           , nlHsDo DoExpr (ss ++ [noLoc $ mkLastStmt b])] +    bindLex pat	       = noLoc (mkBindStmt pat (nlHsVar lexP_RDR))	-- pat <- lexP +    con_app con as     = nlHsVarApps (getRdrName con) as		-- con as +    result_expr con as = nlHsApp (nlHsVar returnM_RDR) (con_app con as) -- return (con as)      punc_pat s   = nlConPat punc_RDR   [nlLitPat (mkHsString s)]  -- Punc 'c' diff --git a/compiler/typecheck/TcHsSyn.lhs b/compiler/typecheck/TcHsSyn.lhs index 06cbe33daf..35da6557fc 100644 --- a/compiler/typecheck/TcHsSyn.lhs +++ b/compiler/typecheck/TcHsSyn.lhs @@ -580,11 +580,10 @@ zonkExpr env (HsLet binds expr)      zonkLExpr new_env expr	`thenM` \ new_expr ->      returnM (HsLet new_binds new_expr) -zonkExpr env (HsDo do_or_lc stmts body ty) -  = zonkStmts env stmts 	`thenM` \ (new_env, new_stmts) -> -    zonkLExpr new_env body	`thenM` \ new_body -> +zonkExpr env (HsDo do_or_lc stmts ty) +  = zonkStmts env stmts 	`thenM` \ (_, new_stmts) ->      zonkTcTypeToType env ty	`thenM` \ new_ty   -> -    returnM (HsDo do_or_lc new_stmts new_body new_ty) +    returnM (HsDo do_or_lc new_stmts new_ty)  zonkExpr env (ExplicitList ty exprs)    = zonkTcTypeToType env ty	`thenM` \ new_ty -> @@ -730,22 +729,26 @@ zonkStmts env (s:ss) = do { (env1, s')  <- wrapLocSndM (zonkStmt env) s  			  ; return (env2, s' : ss') }  zonkStmt :: ZonkEnv -> Stmt TcId -> TcM (ZonkEnv, Stmt Id) -zonkStmt env (ParStmt stmts_w_bndrs) +zonkStmt env (ParStmt stmts_w_bndrs mzip_op bind_op return_op)    = mappM zonk_branch stmts_w_bndrs	`thenM` \ new_stmts_w_bndrs ->      let   	new_binders = concat (map snd new_stmts_w_bndrs)  	env1 = extendZonkEnv env new_binders      in -    return (env1, ParStmt new_stmts_w_bndrs) +    zonkExpr env1 mzip_op   `thenM` \ new_mzip -> +    zonkExpr env1 bind_op   `thenM` \ new_bind -> +    zonkExpr env1 return_op `thenM` \ new_return -> +    return (env1, ParStmt new_stmts_w_bndrs new_mzip new_bind new_return)    where      zonk_branch (stmts, bndrs) = zonkStmts env stmts	`thenM` \ (env1, new_stmts) ->  				 returnM (new_stmts, zonkIdOccs env1 bndrs)  zonkStmt env (RecStmt { recS_stmts = segStmts, recS_later_ids = lvs, recS_rec_ids = rvs                        , recS_ret_fn = ret_id, recS_mfix_fn = mfix_id, recS_bind_fn = bind_id -                      , recS_rec_rets = rets }) +                      , recS_rec_rets = rets, recS_ret_ty = ret_ty })    = do { new_rvs <- zonkIdBndrs env rvs         ; new_lvs <- zonkIdBndrs env lvs +       ; new_ret_ty  <- zonkTcTypeToType env ret_ty         ; new_ret_id  <- zonkExpr env ret_id         ; new_mfix_id <- zonkExpr env mfix_id         ; new_bind_id <- zonkExpr env bind_id @@ -758,28 +761,34 @@ zonkStmt env (RecStmt { recS_stmts = segStmts, recS_later_ids = lvs, recS_rec_id                   RecStmt { recS_stmts = new_segStmts, recS_later_ids = new_lvs                           , recS_rec_ids = new_rvs, recS_ret_fn = new_ret_id                           , recS_mfix_fn = new_mfix_id, recS_bind_fn = new_bind_id -                         , recS_rec_rets = new_rets }) } +                         , recS_rec_rets = new_rets, recS_ret_ty = new_ret_ty }) } -zonkStmt env (ExprStmt expr then_op ty) +zonkStmt env (ExprStmt expr then_op guard_op ty)    = zonkLExpr env expr		`thenM` \ new_expr ->      zonkExpr env then_op	`thenM` \ new_then -> +    zonkExpr env guard_op	`thenM` \ new_guard ->      zonkTcTypeToType env ty	`thenM` \ new_ty -> -    returnM (env, ExprStmt new_expr new_then new_ty) +    returnM (env, ExprStmt new_expr new_then new_guard new_ty) -zonkStmt env (TransformStmt stmts binders usingExpr maybeByExpr) -  = do { (env', stmts') <- zonkStmts env stmts  -    ; let binders' = zonkIdOccs env' binders -    ; usingExpr' <- zonkLExpr env' usingExpr -    ; maybeByExpr' <- zonkMaybeLExpr env' maybeByExpr -    ; return (env', TransformStmt stmts' binders' usingExpr' maybeByExpr') } -     -zonkStmt env (GroupStmt stmts binderMap by using) +zonkStmt env (LastStmt expr ret_op) +  = zonkLExpr env expr		`thenM` \ new_expr -> +    zonkExpr env ret_op		`thenM` \ new_ret -> +    returnM (env, LastStmt new_expr new_ret) + +zonkStmt env (TransStmt { trS_stmts = stmts, trS_bndrs = binderMap +                        , trS_by = by, trS_form = form, trS_using = using +                        , trS_ret = return_op, trS_bind = bind_op, trS_fmap = liftM_op })    = do { (env', stmts') <- zonkStmts env stmts       ; binderMap' <- mappM (zonkBinderMapEntry env') binderMap -    ; by' <- fmapMaybeM (zonkLExpr env') by -    ; using' <- fmapEitherM (zonkLExpr env) (zonkExpr env) using +    ; by'        <- fmapMaybeM (zonkLExpr env') by +    ; using'     <- zonkLExpr env using +    ; return_op' <- zonkExpr env' return_op +    ; bind_op'   <- zonkExpr env' bind_op +    ; liftM_op'  <- zonkExpr env' liftM_op      ; let env'' = extendZonkEnv env' (map snd binderMap') -    ; return (env'', GroupStmt stmts' binderMap' by' using') } +    ; return (env'', TransStmt { trS_stmts = stmts', trS_bndrs = binderMap' +                               , trS_by = by', trS_form = form, trS_using = using' +                               , trS_ret = return_op', trS_bind = bind_op', trS_fmap = liftM_op' }) }    where      zonkBinderMapEntry env (oldBinder, newBinder) = do           let oldBinder' = zonkIdOcc env oldBinder @@ -797,11 +806,6 @@ zonkStmt env (BindStmt pat expr bind_op fail_op)  	; new_fail <- zonkExpr env fail_op  	; return (env1, BindStmt new_pat new_expr new_bind new_fail) } -zonkMaybeLExpr :: ZonkEnv -> Maybe (LHsExpr TcId) -> TcM (Maybe (LHsExpr Id)) -zonkMaybeLExpr _   Nothing  = return Nothing -zonkMaybeLExpr env (Just e) = (zonkLExpr env e) >>= (return . Just) - -  -------------------------------------------------------------------------  zonkRecFields :: ZonkEnv -> HsRecordBinds TcId -> TcM (HsRecordBinds TcId)  zonkRecFields env (HsRecFields flds dd) @@ -1137,4 +1141,4 @@ zonkTcCoToCo env co                                   ; return (mkInstCo co' ty')  }      go (ForAllCo tv co)     = ASSERT( isImmutableTyVar tv )                                do { co' <- go co; return (mkForAllCo tv co') } -\end{code}
\ No newline at end of file +\end{code} diff --git a/compiler/typecheck/TcMatches.lhs b/compiler/typecheck/TcMatches.lhs index f912039be7..ce6c2fc7fb 100644 --- a/compiler/typecheck/TcMatches.lhs +++ b/compiler/typecheck/TcMatches.lhs @@ -6,16 +6,18 @@  TcMatches: Typecheck some @Matches@  \begin{code} +{-# OPTIONS_GHC -w #-}   -- debugging  module TcMatches ( tcMatchesFun, tcGRHSsPat, tcMatchesCase, tcMatchLambda, -		   TcMatchCtxt(..),  -		   tcStmts, tcDoStmts, tcBody, -		   tcDoStmt, tcMDoStmt, tcGuardStmt +		   TcMatchCtxt(..), TcStmtChecker, +		   tcStmts, tcStmtsAndThen, tcDoStmts, tcBody, +		   tcDoStmt, tcGuardStmt         ) where -import {-# SOURCE #-}	TcExpr( tcSyntaxOp, tcInferRhoNC, tcCheckId, +import {-# SOURCE #-}	TcExpr( tcSyntaxOp, tcInferRhoNC, tcInferRho, tcCheckId,                                  tcMonoExpr, tcMonoExprNC, tcPolyExpr )  import HsSyn +import BasicTypes  import TcRnMonad  import TcEnv  import TcPat @@ -28,13 +30,15 @@ import TysWiredIn  import Id  import TyCon  import TysPrim -import Coercion         ( mkSymCo ) +import Coercion         ( isReflCo, mkSymCo )  import Outputable -import BasicTypes	( Arity )  import Util  import SrcLoc  import FastString +-- Create chunkified tuple tybes for monad comprehensions +import MkCore +  import Control.Monad  #include "HsVersions.h" @@ -221,7 +225,7 @@ tcGRHSs ctxt (GRHSs grhss binds) res_ty  tcGRHS :: TcMatchCtxt -> TcRhoType -> GRHS Name -> TcM (GRHS TcId)  tcGRHS ctxt res_ty (GRHS guards rhs) -  = do  { (guards', rhs') <- tcStmts stmt_ctxt tcGuardStmt guards res_ty $ +  = do  { (guards', rhs') <- tcStmtsAndThen stmt_ctxt tcGuardStmt guards res_ty $  			     mc_body ctxt rhs  	; return (GRHS guards' rhs') }    where @@ -238,36 +242,33 @@ tcGRHS ctxt res_ty (GRHS guards rhs)  \begin{code}  tcDoStmts :: HsStmtContext Name   	  -> [LStmt Name] -	  -> LHsExpr Name  	  -> TcRhoType  	  -> TcM (HsExpr TcId)		-- Returns a HsDo -tcDoStmts ListComp stmts body res_ty +tcDoStmts ListComp stmts res_ty    = do	{ (coi, elt_ty) <- matchExpectedListTy res_ty -	; (stmts', body') <- tcStmts ListComp (tcLcStmt listTyCon) stmts  -				     elt_ty $ -			     tcBody body -	; return $ mkHsWrapCo coi  -                     (HsDo ListComp stmts' body' (mkListTy elt_ty)) } +        ; let list_ty = mkListTy elt_ty +	; stmts' <- tcStmts ListComp (tcLcStmt listTyCon) stmts elt_ty +	; return $ mkHsWrapCo coi (HsDo ListComp stmts' list_ty) } -tcDoStmts PArrComp stmts body res_ty +tcDoStmts PArrComp stmts res_ty    = do	{ (coi, elt_ty) <- matchExpectedPArrTy res_ty -	; (stmts', body') <- tcStmts PArrComp (tcLcStmt parrTyCon) stmts  -				     elt_ty $ -			     tcBody body -	; return $ mkHsWrapCo coi  -                     (HsDo PArrComp stmts' body' (mkPArrTy elt_ty)) } +        ; let parr_ty = mkPArrTy elt_ty +	; stmts' <- tcStmts PArrComp (tcLcStmt parrTyCon) stmts elt_ty +	; return $ mkHsWrapCo coi (HsDo PArrComp stmts' parr_ty) } + +tcDoStmts DoExpr stmts res_ty +  = do	{ stmts' <- tcStmts DoExpr tcDoStmt stmts res_ty +	; return (HsDo DoExpr stmts' res_ty) } -tcDoStmts DoExpr stmts body res_ty -  = do	{ (stmts', body') <- tcStmts DoExpr tcDoStmt stmts res_ty $ -			     tcBody body -	; return (HsDo DoExpr stmts' body' res_ty) } +tcDoStmts MDoExpr stmts res_ty +  = do  { stmts' <- tcStmts MDoExpr tcDoStmt stmts res_ty +        ; return (HsDo MDoExpr stmts' res_ty) } -tcDoStmts MDoExpr stmts body res_ty -  = do  { (stmts', body') <- tcStmts MDoExpr tcDoStmt stmts res_ty $ -			     tcBody body -        ; return (HsDo MDoExpr stmts' body' res_ty) } +tcDoStmts MonadComp stmts res_ty +  = do  { stmts' <- tcStmts MonadComp tcMcStmt stmts res_ty  +        ; return (HsDo MonadComp stmts' res_ty) } -tcDoStmts ctxt _ _ _ = pprPanic "tcDoStmts" (pprStmtContext ctxt) +tcDoStmts ctxt _ _ = pprPanic "tcDoStmts" (pprStmtContext ctxt)  tcBody :: LHsExpr Name -> TcRhoType -> TcM (LHsExpr TcId)  tcBody body res_ty @@ -296,40 +297,52 @@ tcStmts :: HsStmtContext Name  	-> TcStmtChecker	-- NB: higher-rank type          -> [LStmt Name]  	-> TcRhoType -	-> (TcRhoType -> TcM thing) -        -> TcM ([LStmt TcId], thing) +        -> TcM [LStmt TcId] +tcStmts ctxt stmt_chk stmts res_ty +  = do { (stmts', _) <- tcStmtsAndThen ctxt stmt_chk stmts res_ty $ +                        const (return ()) +       ; return stmts' } + +tcStmtsAndThen :: HsStmtContext Name +	       -> TcStmtChecker	-- NB: higher-rank type +               -> [LStmt Name] +	       -> TcRhoType +	       -> (TcRhoType -> TcM thing) +               -> TcM ([LStmt TcId], thing)  -- Note the higher-rank type.  stmt_chk is applied at different  -- types in the equations for tcStmts -tcStmts _ _ [] res_ty thing_inside +tcStmtsAndThen _ _ [] res_ty thing_inside    = do	{ thing <- thing_inside res_ty  	; return ([], thing) }  -- LetStmts are handled uniformly, regardless of context -tcStmts ctxt stmt_chk (L loc (LetStmt binds) : stmts) res_ty thing_inside +tcStmtsAndThen ctxt stmt_chk (L loc (LetStmt binds) : stmts) res_ty thing_inside    = do	{ (binds', (stmts',thing)) <- tcLocalBinds binds $ -				      tcStmts ctxt stmt_chk stmts res_ty thing_inside +				      tcStmtsAndThen ctxt stmt_chk stmts res_ty thing_inside  	; return (L loc (LetStmt binds') : stmts', thing) }  -- For the vanilla case, handle the location-setting part -tcStmts ctxt stmt_chk (L loc stmt : stmts) res_ty thing_inside +tcStmtsAndThen ctxt stmt_chk (L loc stmt : stmts) res_ty thing_inside    = do 	{ (stmt', (stmts', thing)) <-  -		setSrcSpan loc		 		$ -    		addErrCtxt (pprStmtInCtxt ctxt stmt)	$ -		stmt_chk ctxt stmt res_ty		$ \ res_ty' -> -		popErrCtxt 				$ -		tcStmts ctxt stmt_chk stmts res_ty'	$ +		setSrcSpan loc		 		    $ +    		addErrCtxt (pprStmtInCtxt ctxt stmt)	    $ +		stmt_chk ctxt stmt res_ty		    $ \ res_ty' -> +		popErrCtxt 				    $ +		tcStmtsAndThen ctxt stmt_chk stmts res_ty'  $  		thing_inside  	; return (L loc stmt' : stmts', thing) } --------------------------------- ---	Pattern guards +--------------------------------------------------- +--	        Pattern guards +--------------------------------------------------- +  tcGuardStmt :: TcStmtChecker -tcGuardStmt _ (ExprStmt guard _ _) res_ty thing_inside +tcGuardStmt _ (ExprStmt guard _ _ _) res_ty thing_inside    = do	{ guard' <- tcMonoExpr guard boolTy  	; thing  <- thing_inside res_ty -	; return (ExprStmt guard' noSyntaxExpr boolTy, thing) } +	; return (ExprStmt guard' noSyntaxExpr noSyntaxExpr boolTy, thing) }  tcGuardStmt ctxt (BindStmt pat rhs _ _) res_ty thing_inside    = do	{ (rhs', rhs_ty) <- tcInferRhoNC rhs	-- Stmt has a context already @@ -341,25 +354,292 @@ tcGuardStmt _ stmt _ _    = pprPanic "tcGuardStmt: unexpected Stmt" (ppr stmt) --------------------------------- ---	List comprehensions and PArrays +--------------------------------------------------- +--	     List comprehensions and PArrays +--	         (no rebindable syntax) +--------------------------------------------------- + +-- Dealt with separately, rather than by tcMcStmt, because +--   a) PArr isn't (yet) an instance of Monad, so the generality seems overkill +--   b) We have special desugaring rules for list comprehensions, +--      which avoid creating intermediate lists.  They in turn  +--      assume that the bind/return operations are the regular +--      polymorphic ones, and in particular don't have any +--      coercion matching stuff in them.  It's hard to avoid the +--      potential for non-trivial coercions in tcMcStmt  tcLcStmt :: TyCon	-- The list/Parray type constructor ([] or PArray)  	 -> TcStmtChecker +tcLcStmt _ _ (LastStmt body _) elt_ty thing_inside +  = do { body' <- tcMonoExprNC body elt_ty +       ; thing <- thing_inside (panic "tcLcStmt: thing_inside") +       ; return (LastStmt body' noSyntaxExpr, thing) } +  -- A generator, pat <- rhs -tcLcStmt m_tc ctxt (BindStmt pat rhs _ _) res_ty thing_inside +tcLcStmt m_tc ctxt (BindStmt pat rhs _ _) elt_ty thing_inside   = do	{ pat_ty <- newFlexiTyVarTy liftedTypeKind          ; rhs'   <- tcMonoExpr rhs (mkTyConApp m_tc [pat_ty])  	; (pat', thing)  <- tcPat (StmtCtxt ctxt) pat pat_ty $ -                            thing_inside res_ty +                            thing_inside elt_ty  	; return (BindStmt pat' rhs' noSyntaxExpr noSyntaxExpr, thing) }  -- A boolean guard -tcLcStmt _ _ (ExprStmt rhs _ _) res_ty thing_inside +tcLcStmt _ _ (ExprStmt rhs _ _ _) elt_ty thing_inside    = do	{ rhs'  <- tcMonoExpr rhs boolTy -	; thing <- thing_inside res_ty -	; return (ExprStmt rhs' noSyntaxExpr boolTy, thing) } +	; thing <- thing_inside elt_ty +	; return (ExprStmt rhs' noSyntaxExpr noSyntaxExpr boolTy, thing) } + +-- ParStmt: See notes with tcMcStmt +tcLcStmt m_tc ctxt (ParStmt bndr_stmts_s _ _ _) elt_ty thing_inside +  = do	{ (pairs', thing) <- loop bndr_stmts_s +	; return (ParStmt pairs' noSyntaxExpr noSyntaxExpr noSyntaxExpr, thing) } +  where +    -- loop :: [([LStmt Name], [Name])] -> TcM ([([LStmt TcId], [TcId])], thing) +    loop [] = do { thing <- thing_inside elt_ty +		 ; return ([], thing) }		-- matching in the branches + +    loop ((stmts, names) : pairs) +      = do { (stmts', (ids, pairs', thing)) +		<- tcStmtsAndThen ctxt (tcLcStmt m_tc) stmts elt_ty $ \ _elt_ty' -> +		   do { ids <- tcLookupLocalIds names +		      ; (pairs', thing) <- loop pairs +		      ; return (ids, pairs', thing) } +	   ; return ( (stmts', ids) : pairs', thing ) } + +tcLcStmt m_tc ctxt (TransStmt { trS_form = form, trS_stmts = stmts +                              , trS_bndrs =  bindersMap +                              , trS_by = by, trS_using = using }) elt_ty thing_inside +  = do { let (bndr_names, n_bndr_names) = unzip bindersMap +             unused_ty = pprPanic "tcLcStmt: inner ty" (ppr bindersMap) +       	     -- The inner 'stmts' lack a LastStmt, so the element type +	     --  passed in to tcStmtsAndThen is never looked at +       ; (stmts', (bndr_ids, by')) +            <- tcStmtsAndThen (TransStmtCtxt ctxt) (tcLcStmt m_tc) stmts unused_ty $ \_ -> do +	       { by' <- case by of +                           Nothing -> return Nothing +                           Just e  -> do { e_ty <- tcInferRho e; return (Just e_ty) } +               ; bndr_ids <- tcLookupLocalIds bndr_names +               ; return (bndr_ids, by') } + +       ; let m_app ty = mkTyConApp m_tc [ty] + +       --------------- Typecheck the 'using' function ------------- +       -- using :: ((a,b,c)->t) -> m (a,b,c) -> m (a,b,c)m      (ThenForm) +       --       :: ((a,b,c)->t) -> m (a,b,c) -> m (m (a,b,c)))  (GroupForm) + +         -- n_app :: Type -> Type   -- Wraps a 'ty' into '[ty]' for GroupForm +       ; let n_app = case form of +                       ThenForm -> (\ty -> ty) +  		       _ 	-> m_app + +             by_arrow :: Type -> Type     -- Wraps 'ty' to '(a->t) -> ty' if the By is present +             by_arrow = case by' of +                          Nothing       -> \ty -> ty +                          Just (_,e_ty) -> \ty -> (alphaTy `mkFunTy` e_ty) `mkFunTy` ty + +             tup_ty        = mkBigCoreVarTupTy bndr_ids +             poly_arg_ty   = m_app alphaTy +	     poly_res_ty   = m_app (n_app alphaTy) +	     using_poly_ty = mkForAllTy alphaTyVar $ by_arrow $  +                             poly_arg_ty `mkFunTy` poly_res_ty + +       ; using' <- tcPolyExpr using using_poly_ty +       ; let final_using = fmap (HsWrap (WpTyApp tup_ty)) using'  + +	     -- 'stmts' returns a result of type (m1_ty tuple_ty), +	     -- typically something like [(Int,Bool,Int)] +	     -- We don't know what tuple_ty is yet, so we use a variable +       ; let mk_n_bndr :: Name -> TcId -> TcId +             mk_n_bndr n_bndr_name bndr_id = mkLocalId n_bndr_name (n_app (idType bndr_id)) + +             -- Ensure that every old binder of type `b` is linked up with its +             -- new binder which should have type `n b` +	     -- See Note [GroupStmt binder map] in HsExpr +             n_bndr_ids  = zipWith mk_n_bndr n_bndr_names bndr_ids +             bindersMap' = bndr_ids `zip` n_bndr_ids + +       -- Type check the thing in the environment with  +       -- these new binders and return the result +       ; thing <- tcExtendIdEnv n_bndr_ids (thing_inside elt_ty) + +       ; return (emptyTransStmt { trS_stmts = stmts', trS_bndrs = bindersMap'  +                                , trS_by = fmap fst by', trS_using = final_using  +                                , trS_form = form }, thing) } +     +tcLcStmt _ _ stmt _ _ +  = pprPanic "tcLcStmt: unexpected Stmt" (ppr stmt) + + +--------------------------------------------------- +--	     Monad comprehensions  +--	  (supports rebindable syntax) +--------------------------------------------------- + +tcMcStmt :: TcStmtChecker + +tcMcStmt _ (LastStmt body return_op) res_ty thing_inside +  = do  { a_ty       <- newFlexiTyVarTy liftedTypeKind +        ; return_op' <- tcSyntaxOp MCompOrigin return_op +                                   (a_ty `mkFunTy` res_ty) +        ; body'      <- tcMonoExprNC body a_ty +        ; thing      <- thing_inside (panic "tcMcStmt: thing_inside") +        ; return (LastStmt body' return_op', thing) }  + +-- Generators for monad comprehensions ( pat <- rhs ) +-- +--   [ body | q <- gen ]  ->  gen :: m a +--                            q   ::   a +-- + +tcMcStmt ctxt (BindStmt pat rhs bind_op fail_op) res_ty thing_inside + = do   { rhs_ty     <- newFlexiTyVarTy liftedTypeKind +        ; pat_ty     <- newFlexiTyVarTy liftedTypeKind +        ; new_res_ty <- newFlexiTyVarTy liftedTypeKind + +	   -- (>>=) :: rhs_ty -> (pat_ty -> new_res_ty) -> res_ty +        ; bind_op'   <- tcSyntaxOp MCompOrigin bind_op  +                             (mkFunTys [rhs_ty, mkFunTy pat_ty new_res_ty] res_ty) + +           -- If (but only if) the pattern can fail, typecheck the 'fail' operator +        ; fail_op' <- if isIrrefutableHsPat pat  +                      then return noSyntaxExpr +                      else tcSyntaxOp MCompOrigin fail_op (mkFunTy stringTy new_res_ty) + +        ; rhs' <- tcMonoExprNC rhs rhs_ty +        ; (pat', thing) <- tcPat (StmtCtxt ctxt) pat pat_ty $ +                           thing_inside new_res_ty + +        ; return (BindStmt pat' rhs' bind_op' fail_op', thing) } + +-- Boolean expressions. +-- +--   [ body | stmts, expr ]  ->  expr :: m Bool +-- +tcMcStmt _ (ExprStmt rhs then_op guard_op _) res_ty thing_inside +  = do	{ -- Deal with rebindable syntax: +          --    guard_op :: test_ty -> rhs_ty +          --    then_op  :: rhs_ty -> new_res_ty -> res_ty +          -- Where test_ty is, for example, Bool +          test_ty    <- newFlexiTyVarTy liftedTypeKind +        ; rhs_ty     <- newFlexiTyVarTy liftedTypeKind +        ; new_res_ty <- newFlexiTyVarTy liftedTypeKind +        ; rhs'       <- tcMonoExpr rhs test_ty +        ; guard_op'  <- tcSyntaxOp MCompOrigin guard_op +                                   (mkFunTy test_ty rhs_ty) +        ; then_op'   <- tcSyntaxOp MCompOrigin then_op +		                   (mkFunTys [rhs_ty, new_res_ty] res_ty) +	; thing      <- thing_inside new_res_ty +	; return (ExprStmt rhs' then_op' guard_op' rhs_ty, thing) } + +-- Grouping statements +-- +--   [ body | stmts, then group by e ] +--     ->  e :: t +--   [ body | stmts, then group by e using f ] +--     ->  e :: t +--         f :: forall a. (a -> t) -> m a -> m (m a) +--   [ body | stmts, then group using f ] +--     ->  f :: forall a. m a -> m (m a) + +-- We type [ body | (stmts, group by e using f), ... ] +--     f <optional by> [ (a,b,c) | stmts ] >>= \(a,b,c) -> ...body.... +-- +-- We type the functions as follows: +--     f <optional by> :: m1 (a,b,c) -> m2 (a,b,c)		(ThenForm) +--     	 	       :: m1 (a,b,c) -> m2 (n (a,b,c))		(GroupForm) +--     (>>=) :: m2 (a,b,c)     -> ((a,b,c)   -> res) -> res	(ThenForm) +--           :: m2 (n (a,b,c)) -> (n (a,b,c) -> res) -> res	(GroupForm) +--  +tcMcStmt ctxt (TransStmt { trS_stmts = stmts, trS_bndrs = bindersMap +                         , trS_by = by, trS_using = using, trS_form = form +                         , trS_ret = return_op, trS_bind = bind_op  +                         , trS_fmap = fmap_op }) res_ty thing_inside +  = do { let star_star_kind = liftedTypeKind `mkArrowKind` liftedTypeKind +       ; m1_ty   <- newFlexiTyVarTy star_star_kind +       ; m2_ty   <- newFlexiTyVarTy star_star_kind +       ; tup_ty  <- newFlexiTyVarTy liftedTypeKind +       ; by_e_ty <- newFlexiTyVarTy liftedTypeKind  -- The type of the 'by' expression (if any) + +         -- n_app :: Type -> Type   -- Wraps a 'ty' into '(n ty)' for GroupForm +       ; n_app <- case form of +                    ThenForm -> return (\ty -> ty) +		    _ 	     -> do { n_ty <- newFlexiTyVarTy star_star_kind +                      	           ; return (n_ty `mkAppTy`) } +       ; let by_arrow :: Type -> Type      +             -- (by_arrow res) produces ((alpha->e_ty) -> res)     ('by' present) +             --                          or res                    ('by' absent)  +             by_arrow = case by of +                          Nothing -> \res -> res +                          Just {} -> \res -> (alphaTy `mkFunTy` by_e_ty) `mkFunTy` res + +             poly_arg_ty  = m1_ty `mkAppTy` alphaTy +             using_arg_ty = m1_ty `mkAppTy` tup_ty +	     poly_res_ty  = m2_ty `mkAppTy` n_app alphaTy +	     using_res_ty = m2_ty `mkAppTy` n_app tup_ty +	     using_poly_ty = mkForAllTy alphaTyVar $ by_arrow $  +                             poly_arg_ty `mkFunTy` poly_res_ty + +	     -- 'stmts' returns a result of type (m1_ty tuple_ty), +	     -- typically something like [(Int,Bool,Int)] +	     -- We don't know what tuple_ty is yet, so we use a variable +       ; let (bndr_names, n_bndr_names) = unzip bindersMap +       ; (stmts', (bndr_ids, by', return_op')) <- +            tcStmtsAndThen (TransStmtCtxt ctxt) tcMcStmt stmts using_arg_ty $ \res_ty' -> do +	        { by' <- case by of +                           Nothing -> return Nothing +                           Just e  -> do { e' <- tcMonoExpr e by_e_ty; return (Just e') } + +                -- Find the Ids (and hence types) of all old binders +                ; bndr_ids <- tcLookupLocalIds bndr_names + +                -- 'return' is only used for the binders, so we know its type. +                --   return :: (a,b,c,..) -> m (a,b,c,..) +                ; return_op' <- tcSyntaxOp MCompOrigin return_op $  +                                (mkBigCoreVarTupTy bndr_ids) `mkFunTy` res_ty' + +                ; return (bndr_ids, by', return_op') } + +       --------------- Typecheck the 'bind' function ------------- +       -- (>>=) :: m2 (n (a,b,c)) -> ( n (a,b,c) -> new_res_ty ) -> res_ty +       ; new_res_ty <- newFlexiTyVarTy liftedTypeKind +       ; bind_op' <- tcSyntaxOp MCompOrigin bind_op $ +                                using_res_ty `mkFunTy` (n_app tup_ty `mkFunTy` new_res_ty) +                                             `mkFunTy` res_ty + +       --------------- Typecheck the 'fmap' function ------------- +       ; fmap_op' <- case form of +                       ThenForm -> return noSyntaxExpr +                       _ -> fmap unLoc . tcPolyExpr (noLoc fmap_op) $ +                            mkForAllTy alphaTyVar $ mkForAllTy betaTyVar $ +                            (alphaTy `mkFunTy` betaTy) +                            `mkFunTy` (n_app alphaTy) +                            `mkFunTy` (n_app betaTy) + +       --------------- Typecheck the 'using' function ------------- +       -- using :: ((a,b,c)->t) -> m1 (a,b,c) -> m2 (n (a,b,c)) + +       ; using' <- tcPolyExpr using using_poly_ty +       ; let final_using = fmap (HsWrap (WpTyApp tup_ty)) using'  + +       --------------- Bulding the bindersMap ---------------- +       ; let mk_n_bndr :: Name -> TcId -> TcId +             mk_n_bndr n_bndr_name bndr_id = mkLocalId n_bndr_name (n_app (idType bndr_id)) + +             -- Ensure that every old binder of type `b` is linked up with its +             -- new binder which should have type `n b` +	     -- See Note [GroupStmt binder map] in HsExpr +             n_bndr_ids = zipWith mk_n_bndr n_bndr_names bndr_ids +             bindersMap' = bndr_ids `zip` n_bndr_ids + +       -- Type check the thing in the environment with  +       -- these new binders and return the result +       ; thing <- tcExtendIdEnv n_bndr_ids (thing_inside new_res_ty) + +       ; return (TransStmt { trS_stmts = stmts', trS_bndrs = bindersMap'  +                           , trS_by = by', trS_using = final_using  +                           , trS_ret = return_op', trS_bind = bind_op' +                           , trS_fmap = fmap_op', trS_form = form }, thing) }  -- A parallel set of comprehensions  --	[ (g x, h x) | ... ; let g v = ... @@ -381,106 +661,95 @@ tcLcStmt _ _ (ExprStmt rhs _ _) res_ty thing_inside  -- ensure that g,h and x,y don't duplicate, and simply grow the environment.  -- So the binders of the first parallel group will be in scope in the second  -- group.  But that's fine; there's no shadowing to worry about. - -tcLcStmt m_tc ctxt (ParStmt bndr_stmts_s) elt_ty thing_inside -  = do	{ (pairs', thing) <- loop bndr_stmts_s -	; return (ParStmt pairs', thing) } -  where -    -- loop :: [([LStmt Name], [Name])] -> TcM ([([LStmt TcId], [TcId])], thing) -    loop [] = do { thing <- thing_inside elt_ty -		 ; return ([], thing) }		-- matching in the branches - -    loop ((stmts, names) : pairs) -      = do { (stmts', (ids, pairs', thing)) -		<- tcStmts ctxt (tcLcStmt m_tc) stmts elt_ty $ \ _elt_ty' -> -		   do { ids <- tcLookupLocalIds names -		      ; (pairs', thing) <- loop pairs -		      ; return (ids, pairs', thing) } -	   ; return ( (stmts', ids) : pairs', thing ) } - -tcLcStmt m_tc ctxt (TransformStmt stmts binders usingExpr maybeByExpr) elt_ty thing_inside = do -    (stmts', (binders', usingExpr', maybeByExpr', thing)) <-  -        tcStmts (TransformStmtCtxt ctxt) (tcLcStmt m_tc) stmts elt_ty $ \elt_ty' -> do -            let alphaListTy = mkTyConApp m_tc [alphaTy] -                     -            (usingExpr', maybeByExpr') <-  -                case maybeByExpr of -                    Nothing -> do -                        -- We must validate that usingExpr :: forall a. [a] -> [a] -                        let using_ty = mkForAllTy alphaTyVar (alphaListTy `mkFunTy` alphaListTy) -                        usingExpr' <- tcPolyExpr usingExpr using_ty -                        return (usingExpr', Nothing) -                    Just byExpr -> do -                        -- We must infer a type such that e :: t and then check that  -			-- usingExpr :: forall a. (a -> t) -> [a] -> [a] -                        (byExpr', tTy) <- tcInferRhoNC byExpr -                        let using_ty = mkForAllTy alphaTyVar $  -                                       (alphaTy `mkFunTy` tTy) -                                       `mkFunTy` alphaListTy `mkFunTy` alphaListTy -                        usingExpr' <- tcPolyExpr usingExpr using_ty -                        return (usingExpr', Just byExpr') -             -            binders' <- tcLookupLocalIds binders -            thing <- thing_inside elt_ty' -             -            return (binders', usingExpr', maybeByExpr', thing) - -    return (TransformStmt stmts' binders' usingExpr' maybeByExpr', thing) - -tcLcStmt m_tc ctxt (GroupStmt stmts bindersMap by using) elt_ty thing_inside -  = do { let (bndr_names, list_bndr_names) = unzip bindersMap - -       ; (stmts', (bndr_ids, by', using_ty, elt_ty')) <- -            tcStmts (TransformStmtCtxt ctxt) (tcLcStmt m_tc) stmts elt_ty $ \elt_ty' -> do -	        (by', using_ty) <-  -                   case by of -                     Nothing   -> -- check that using :: forall a. [a] -> [[a]] -                                  return (Nothing, mkForAllTy alphaTyVar $ -                                                   alphaListTy `mkFunTy` alphaListListTy) -		     			 -		     Just by_e -> -- check that using :: forall a. (a -> t) -> [a] -> [[a]] -		     	          -- where by :: t -                                  do { (by_e', t_ty) <- tcInferRhoNC by_e -                                     ; return (Just by_e', mkForAllTy alphaTyVar $ -                                                           (alphaTy `mkFunTy` t_ty)  -                                                           `mkFunTy` alphaListTy  -                                                           `mkFunTy` alphaListListTy) } -                -- Find the Ids (and hence types) of all old binders -                bndr_ids <- tcLookupLocalIds bndr_names -                 -                return (bndr_ids, by', using_ty, elt_ty') -         -                -- Ensure that every old binder of type b is linked up with -		-- its new binder which should have type [b] -       ; let list_bndr_ids = zipWith mk_list_bndr list_bndr_names bndr_ids -             bindersMap' = bndr_ids `zip` list_bndr_ids -	     -- See Note [GroupStmt binder map] in HsExpr -             -       ; using' <- case using of -                     Left  e -> do { e' <- tcPolyExpr e         using_ty; return (Left  e') } -                     Right e -> do { e' <- tcPolyExpr (noLoc e) using_ty; return (Right (unLoc e')) } - -             -- Type check the thing in the environment with  -	     -- these new binders and return the result -       ; thing <- tcExtendIdEnv list_bndr_ids (thing_inside elt_ty') -       ; return (GroupStmt stmts' bindersMap' by' using', thing) } -  where -    alphaListTy = mkTyConApp m_tc [alphaTy] -    alphaListListTy = mkTyConApp m_tc [alphaListTy] -             -    mk_list_bndr :: Name -> TcId -> TcId -    mk_list_bndr list_bndr_name bndr_id  -      = mkLocalId list_bndr_name (mkTyConApp m_tc [idType bndr_id]) -     -tcLcStmt _ _ stmt _ _ -  = pprPanic "tcLcStmt: unexpected Stmt" (ppr stmt) -         --------------------------------- ---	Do-notation --- The main excitement here is dealing with rebindable syntax +-- +-- Note: The `mzip` function will get typechecked via: +-- +--   ParStmt [st1::t1, st2::t2, st3::t3] +--    +--   mzip :: m st1 +--        -> (m st2 -> m st3 -> m (st2, st3))   -- recursive call +--        -> m (st1, (st2, st3)) +-- +tcMcStmt ctxt (ParStmt bndr_stmts_s mzip_op bind_op return_op) res_ty thing_inside +  = do { let star_star_kind = liftedTypeKind `mkArrowKind` liftedTypeKind +       ; m_ty   <- newFlexiTyVarTy star_star_kind + +       ; let mzip_ty  = mkForAllTys [alphaTyVar, betaTyVar] $ +                        (m_ty `mkAppTy` alphaTy) +                        `mkFunTy` +                        (m_ty `mkAppTy` betaTy) +                        `mkFunTy` +                        (m_ty `mkAppTy` mkBoxedTupleTy [alphaTy, betaTy]) +       ; mzip_op' <- unLoc `fmap` tcPolyExpr (noLoc mzip_op) mzip_ty + +       ; return_op' <- fmap unLoc . tcPolyExpr (noLoc return_op) $ +                       mkForAllTy alphaTyVar $ +                       alphaTy `mkFunTy` (m_ty `mkAppTy` alphaTy) + +       ; (pairs', thing) <- loop m_ty bndr_stmts_s + +       -- Typecheck bind: +       ; let tys      = map (mkBigCoreVarTupTy . snd) pairs' +             tuple_ty = mk_tuple_ty tys + +       ; bind_op' <- tcSyntaxOp MCompOrigin bind_op $ +                        (m_ty `mkAppTy` tuple_ty) +                        `mkFunTy` (tuple_ty `mkFunTy` res_ty) +                        `mkFunTy` res_ty + +       ; return (ParStmt pairs' mzip_op' bind_op' return_op', thing) } + +  where  +    mk_tuple_ty tys = foldr1 (\tn tm -> mkBoxedTupleTy [tn, tm]) tys + +       -- loop :: Type                                  -- m_ty +       --      -> [([LStmt Name], [Name])] +       --      -> TcM ([([LStmt TcId], [TcId])], thing) +    loop _ [] = do { thing <- thing_inside res_ty +                   ; return ([], thing) }           -- matching in the branches + +    loop m_ty ((stmts, names) : pairs) +      = do { -- type dummy since we don't know all binder types yet +             ty_dummy <- newFlexiTyVarTy liftedTypeKind +           ; (stmts', (ids, pairs', thing)) +                <- tcStmtsAndThen ctxt tcMcStmt stmts ty_dummy $ \res_ty' -> +                   do { ids <- tcLookupLocalIds names +    		      ; let m_tup_ty = m_ty `mkAppTy` mkBigCoreVarTupTy ids + +    		      ; check_same m_tup_ty res_ty' +    		      ; check_same m_tup_ty ty_dummy +    							  +                      ; (pairs', thing) <- loop m_ty pairs +                      ; return (ids, pairs', thing) } +           ; return ( (stmts', ids) : pairs', thing ) } + +	-- Check that the types match up. +	-- This is a grevious hack.  They always *will* match  +	-- If (>>=) and (>>) are polymorpic in the return type, +	-- but we don't have any good way to incorporate the coercion +	-- so for now we just check that it's the identity +    check_same actual expected +      = do { coi <- unifyType actual expected +	   ; unless (isIdentityCoI coi) $ +             failWithMisMatch [UnifyOrigin { uo_expected = expected +                                           , uo_actual = actual }] } + +tcMcStmt _ stmt _ _ +  = pprPanic "tcMcStmt: unexpected Stmt" (ppr stmt) + + +--------------------------------------------------- +--	     Do-notation +--	  (supports rebindable syntax) +---------------------------------------------------  tcDoStmt :: TcStmtChecker +tcDoStmt _ (LastStmt body _) res_ty thing_inside +  = do { body' <- tcMonoExprNC body res_ty +       ; thing <- thing_inside (panic "tcDoStmt: thing_inside") +       ; return (LastStmt body' noSyntaxExpr, thing) } +  tcDoStmt ctxt (BindStmt pat rhs bind_op fail_op) res_ty thing_inside    = do	{ 	-- Deal with rebindable syntax:  		--	 (>>=) :: rhs_ty -> (pat_ty -> new_res_ty) -> res_ty @@ -510,7 +779,7 @@ tcDoStmt ctxt (BindStmt pat rhs bind_op fail_op) res_ty thing_inside  	; return (BindStmt pat' rhs' bind_op' fail_op', thing) } -tcDoStmt _ (ExprStmt rhs then_op _) res_ty thing_inside +tcDoStmt _ (ExprStmt rhs then_op _ _) res_ty thing_inside    = do	{   	-- Deal with rebindable syntax;                   --   (>>) :: rhs_ty -> new_res_ty -> res_ty  		-- See also Note [Treat rebindable syntax first] @@ -521,7 +790,7 @@ tcDoStmt _ (ExprStmt rhs then_op _) res_ty thing_inside          ; rhs' <- tcMonoExprNC rhs rhs_ty  	; thing <- thing_inside new_res_ty -	; return (ExprStmt rhs' then_op' rhs_ty, thing) } +	; return (ExprStmt rhs' then_op' noSyntaxExpr rhs_ty, thing) }  tcDoStmt ctxt (RecStmt { recS_stmts = stmts, recS_later_ids = later_names                         , recS_rec_ids = rec_names, recS_ret_fn = ret_op @@ -535,7 +804,7 @@ tcDoStmt ctxt (RecStmt { recS_stmts = stmts, recS_later_ids = later_names          ; tcExtendIdEnv tup_ids $ do          { stmts_ty <- newFlexiTyVarTy liftedTypeKind          ; (stmts', (ret_op', tup_rets)) -                <- tcStmts ctxt tcDoStmt stmts stmts_ty   $ \ inner_res_ty -> +                <- tcStmtsAndThen ctxt tcDoStmt stmts stmts_ty   $ \ inner_res_ty ->                     do { tup_rets <- zipWithM tcCheckId tup_names tup_elt_tys                               -- Unify the types of the "final" Ids (which may                                -- be polymorphic) with those of "knot-tied" Ids @@ -551,7 +820,6 @@ tcDoStmt ctxt (RecStmt { recS_stmts = stmts, recS_later_ids = later_names  			         (mkFunTys [mfix_res_ty, mkFunTy tup_ty new_res_ty] res_ty)          ; thing <- thing_inside new_res_ty ---         ; lie_binds <- bindLocalMethods lie tup_ids          ; let rec_ids = takeList rec_names tup_ids  	; later_ids <- tcLookupLocalIds later_names @@ -560,7 +828,7 @@ tcDoStmt ctxt (RecStmt { recS_stmts = stmts, recS_later_ids = later_names          ; return (RecStmt { recS_stmts = stmts', recS_later_ids = later_ids                            , recS_rec_ids = rec_ids, recS_ret_fn = ret_op'                             , recS_mfix_fn = mfix_op', recS_bind_fn = bind_op' -                          , recS_rec_rets = tup_rets }, thing) +                          , recS_rec_rets = tup_rets, recS_ret_ty = stmts_ty }, thing)          }}  tcDoStmt _ stmt _ _ @@ -577,51 +845,6 @@ rebindable syntax first, and push that information into (tcMonoExprNC rhs).  Otherwise the error shows up when cheking the rebindable syntax, and  the expected/inferred stuff is back to front (see Trac #3613). -\begin{code} --------------------------------- ---	Mdo-notation --- The distinctive features here are ---	(a) RecStmts, and ---	(b) no rebindable syntax - -tcMDoStmt :: (LHsExpr Name -> TcM (LHsExpr TcId, TcType))	-- RHS inference -	  -> TcStmtChecker -tcMDoStmt tc_rhs ctxt (BindStmt pat rhs _ _) res_ty thing_inside -  = do	{ (rhs', pat_ty) <- tc_rhs rhs -	; (pat', thing)  <- tcPat (StmtCtxt ctxt) pat pat_ty $ -                            thing_inside res_ty -	; return (BindStmt pat' rhs' noSyntaxExpr noSyntaxExpr, thing) } - -tcMDoStmt tc_rhs _ (ExprStmt rhs _ _) res_ty thing_inside -  = do	{ (rhs', elt_ty) <- tc_rhs rhs -	; thing 	 <- thing_inside res_ty -	; return (ExprStmt rhs' noSyntaxExpr elt_ty, thing) } - -tcMDoStmt tc_rhs ctxt (RecStmt { recS_stmts = stmts, recS_later_ids = laterNames -                               , recS_rec_ids = recNames }) res_ty thing_inside -  = do	{ rec_tys <- newFlexiTyVarTys (length recNames) liftedTypeKind -	; let rec_ids = zipWith mkLocalId recNames rec_tys -	; tcExtendIdEnv rec_ids			$ do -    	{ (stmts', (later_ids, rec_rets)) -		<- tcStmts ctxt (tcMDoStmt tc_rhs) stmts res_ty	$ \ _res_ty' -> -			-- ToDo: res_ty not really right -		   do { rec_rets <- zipWithM tcCheckId recNames rec_tys -		      ; later_ids <- tcLookupLocalIds laterNames -		      ; return (later_ids, rec_rets) } - -	; thing <- tcExtendIdEnv later_ids (thing_inside res_ty) -		-- NB:	The rec_ids for the recursive things  -		-- 	already scope over this part. This binding may shadow -		--	some of them with polymorphic things with the same Name -		--	(see note [RecStmt] in HsExpr) - -        ; return (RecStmt stmts' later_ids rec_ids noSyntaxExpr noSyntaxExpr noSyntaxExpr rec_rets, thing) -	}} - -tcMDoStmt _ _ stmt _ _ -  = pprPanic "tcMDoStmt: unexpected Stmt" (ppr stmt) -\end{code} -  %************************************************************************  %*									* diff --git a/compiler/typecheck/TcPat.lhs b/compiler/typecheck/TcPat.lhs index c860bfe808..7d725d7020 100644 --- a/compiler/typecheck/TcPat.lhs +++ b/compiler/typecheck/TcPat.lhs @@ -36,7 +36,6 @@ import PrelNames  import BasicTypes hiding (SuccessFlag(..))  import DynFlags  import SrcLoc -import ErrUtils  import Util  import Outputable  import FastString @@ -348,9 +347,9 @@ tc_lpat :: LPat Name  	-> TcM a  	-> TcM (LPat TcId, a)  tc_lpat (L span pat) pat_ty penv thing_inside -  = setSrcSpan span		  $ -    maybeAddErrCtxt (patCtxt pat) $ -    do	{ (pat', res) <- tc_pat penv pat pat_ty thing_inside +  = setSrcSpan span $ +    do	{ (pat', res) <- maybeWrapPatCtxt pat (tc_pat penv pat pat_ty) +                                          thing_inside  	; return (L span pat', res) }  tc_lpats :: PatEnv @@ -772,7 +771,6 @@ matchExpectedConTy data_tc pat_ty         	     -- coi : T tys ~ pat_ty  \end{code} -Noate [  Note [Matching constructor patterns]  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~  Suppose (coi, tys) = matchExpectedConType data_tc pat_ty @@ -1004,12 +1002,18 @@ sigPatCtxt pats bound_tvs pat_tys body_ty tidy_env  -}  \begin{code} -patCtxt :: Pat Name -> Maybe Message	-- Not all patterns are worth pushing a context -patCtxt (VarPat _)  = Nothing -patCtxt (ParPat _)  = Nothing -patCtxt (AsPat _ _) = Nothing -patCtxt pat 	    = Just (hang (ptext (sLit "In the pattern:"))  -                         2 (ppr pat)) +maybeWrapPatCtxt :: Pat Name -> (TcM a -> TcM b) -> TcM a -> TcM b +-- Not all patterns are worth pushing a context +maybeWrapPatCtxt pat tcm thing_inside  +  | not (worth_wrapping pat) = tcm thing_inside +  | otherwise                = addErrCtxt msg $ tcm $ popErrCtxt thing_inside +    			       -- Remember to pop before doing thing_inside +  where +   worth_wrapping (VarPat {}) = False +   worth_wrapping (ParPat {}) = False +   worth_wrapping (AsPat {})  = False +   worth_wrapping _  	      = True +   msg = hang (ptext (sLit "In the pattern:")) 2 (ppr pat)  -----------------------------------------------  checkExistentials :: [TyVar] -> PatEnv -> TcM () diff --git a/compiler/typecheck/TcRnDriver.lhs b/compiler/typecheck/TcRnDriver.lhs index e2c79ee393..9bfde666d3 100644 --- a/compiler/typecheck/TcRnDriver.lhs +++ b/compiler/typecheck/TcRnDriver.lhs @@ -1195,7 +1195,7 @@ runPlans (p:ps) = tryTcLIE_ (runPlans ps) p  --------------------  mkPlan :: LStmt Name -> TcM PlanResult -mkPlan (L loc (ExprStmt expr _ _))	-- An expression typed at the prompt  +mkPlan (L loc (ExprStmt expr _ _ _))	-- An expression typed at the prompt     = do	{ uniq <- newUnique		-- is treated very specially  	; let fresh_it  = itName uniq  	      the_bind  = L loc $ mkFunBind (L loc fresh_it) matches @@ -1204,7 +1204,7 @@ mkPlan (L loc (ExprStmt expr _ _))	-- An expression typed at the prompt  	      bind_stmt = L loc $ BindStmt (nlVarPat fresh_it) expr  					   (HsVar bindIOName) noSyntaxExpr   	      print_it  = L loc $ ExprStmt (nlHsApp (nlHsVar printName) (nlHsVar fresh_it)) -			          	   (HsVar thenIOName) placeHolderType +			          	   (HsVar thenIOName) noSyntaxExpr placeHolderType  	-- The plans are:  	--	[it <- e; print it]	but not if it::() @@ -1232,7 +1232,7 @@ mkPlan (L loc (ExprStmt expr _ _))	-- An expression typed at the prompt  mkPlan stmt@(L loc (BindStmt {}))    | [v] <- collectLStmtBinders stmt		-- One binder, for a bind stmt     = do	{ let print_v  = L loc $ ExprStmt (nlHsApp (nlHsVar printName) (nlHsVar v)) -			          	   (HsVar thenIOName) placeHolderType +			          	  (HsVar thenIOName) noSyntaxExpr placeHolderType  	; print_bind_result <- doptM Opt_PrintBindResult  	; let print_plan = do @@ -1259,11 +1259,25 @@ tcGhciStmts stmts  	let {  	    ret_ty    = mkListTy unitTy ;  	    io_ret_ty = mkTyConApp ioTyCon [ret_ty] ; -	    tc_io_stmts stmts = tcStmts GhciStmt tcDoStmt stmts io_ret_ty ; - +	    tc_io_stmts stmts = tcStmtsAndThen GhciStmt tcDoStmt stmts io_ret_ty ;  	    names = collectLStmtsBinders stmts ; +	 } ; + +	-- OK, we're ready to typecheck the stmts +	traceTc "TcRnDriver.tcGhciStmts: tc stmts" empty ; +	((tc_stmts, ids), lie) <- captureConstraints $  +                                  tc_io_stmts stmts  $ \ _ -> +                           	  mapM tcLookupId names  ; +			-- Look up the names right in the middle, +			-- where they will all be in scope -		-- mk_return builds the expression +	-- Simplify the context +	traceTc "TcRnDriver.tcGhciStmts: simplify ctxt" empty ; +	const_binds <- checkNoErrs (simplifyInteractive lie) ; +		-- checkNoErrs ensures that the plan fails if context redn fails + +	traceTc "TcRnDriver.tcGhciStmts: done" empty ; +        let {   -- mk_return builds the expression  		--	returnIO @ [()] [coerce () x, ..,  coerce () z]  		--  		-- Despite the inconvenience of building the type applications etc, @@ -1274,27 +1288,14 @@ tcGhciStmts stmts  		-- then the type checker would instantiate x..z, and we wouldn't  		-- get their *polymorphic* values.  (And we'd get ambiguity errs  		-- if they were overloaded, since they aren't applied to anything.) -	    mk_return ids = nlHsApp (nlHsTyApp ret_id [ret_ty])  -			 	    (noLoc $ ExplicitList unitTy (map mk_item ids)) ; +	    ret_expr = nlHsApp (nlHsTyApp ret_id [ret_ty])  +		       (noLoc $ ExplicitList unitTy (map mk_item ids)) ;  	    mk_item id = nlHsApp (nlHsTyApp unsafeCoerceId [idType id, unitTy]) -		    	         (nlHsVar id)  -	 } ; - -	-- OK, we're ready to typecheck the stmts -	traceTc "TcRnDriver.tcGhciStmts: tc stmts" empty ; -	((tc_stmts, ids), lie) <- captureConstraints $ tc_io_stmts stmts $ \ _ -> -					   mapM tcLookupId names ; -					-- Look up the names right in the middle, -					-- where they will all be in scope - -	-- Simplify the context -	traceTc "TcRnDriver.tcGhciStmts: simplify ctxt" empty ; -	const_binds <- checkNoErrs (simplifyInteractive lie) ; -		-- checkNoErrs ensures that the plan fails if context redn fails - -	traceTc "TcRnDriver.tcGhciStmts: done" empty ; +		    	         (nlHsVar id) ; +	    stmts = tc_stmts ++ [noLoc (mkLastStmt ret_expr)] +        } ;  	return (ids, mkHsDictLet (EvBinds const_binds) $ -		     noLoc (HsDo GhciStmt tc_stmts (mk_return ids) io_ret_ty)) +		     noLoc (HsDo GhciStmt stmts io_ret_ty))      }  \end{code} diff --git a/compiler/typecheck/TcRnMonad.lhs b/compiler/typecheck/TcRnMonad.lhs index 9193eb5ea0..1b8dd0f5c8 100644 --- a/compiler/typecheck/TcRnMonad.lhs +++ b/compiler/typecheck/TcRnMonad.lhs @@ -780,11 +780,6 @@ updCtxt :: ([ErrCtxt] -> [ErrCtxt]) -> TcM a -> TcM a  updCtxt upd = updLclEnv (\ env@(TcLclEnv { tcl_ctxt = ctxt }) ->   			   env { tcl_ctxt = upd ctxt }) --- Conditionally add an error context -maybeAddErrCtxt :: Maybe Message -> TcM a -> TcM a -maybeAddErrCtxt (Just msg) thing_inside = addErrCtxt msg thing_inside -maybeAddErrCtxt Nothing    thing_inside = thing_inside -  popErrCtxt :: TcM a -> TcM a  popErrCtxt = updCtxt (\ msgs -> case msgs of { [] -> []; (_ : ms) -> ms }) diff --git a/compiler/typecheck/TcRnTypes.lhs b/compiler/typecheck/TcRnTypes.lhs index 607637a96e..7d761eb9e0 100644 --- a/compiler/typecheck/TcRnTypes.lhs +++ b/compiler/typecheck/TcRnTypes.lhs @@ -1113,6 +1113,7 @@ data CtOrigin    | StandAloneDerivOrigin -- Typechecking stand-alone deriving    | DefaultOrigin	-- Typechecking a default decl    | DoOrigin		-- Arising from a do expression +  | MCompOrigin         -- Arising from a monad comprehension    | IfOrigin            -- Arising from an if statement    | ProcOrigin		-- Arising from a proc expression    | AnnOrigin           -- An annotation @@ -1148,6 +1149,7 @@ pprO DerivOrigin	   = ptext (sLit "the 'deriving' clause of a data type declarat  pprO StandAloneDerivOrigin = ptext (sLit "a 'deriving' declaration")  pprO DefaultOrigin	   = ptext (sLit "a 'default' declaration")  pprO DoOrigin	           = ptext (sLit "a do statement") +pprO MCompOrigin           = ptext (sLit "a statement in a monad comprehension")  pprO ProcOrigin	           = ptext (sLit "a proc expression")  pprO (TypeEqOrigin eq)     = ptext (sLit "an equality") <+> ppr eq  pprO AnnOrigin             = ptext (sLit "an annotation") diff --git a/compiler/typecheck/TcSMonad.lhs b/compiler/typecheck/TcSMonad.lhs index dfaa3dc11a..b2ce381707 100644 --- a/compiler/typecheck/TcSMonad.lhs +++ b/compiler/typecheck/TcSMonad.lhs @@ -103,12 +103,13 @@ import FastString  import HsBinds               -- for TcEvBinds stuff   import Id  -  import TcRnTypes +import Data.IORef +  #ifdef DEBUG +import StaticFlags( opt_PprStyle_Debug )  import Control.Monad( when )  #endif -import Data.IORef  \end{code} @@ -529,7 +530,7 @@ runTcS context untouch tcs  #ifdef DEBUG         ; count <- TcM.readTcRef step_count -       ; when (count > 0) $ +       ; when (opt_PprStyle_Debug && count > 0) $           TcM.debugDumpTcRn (ptext (sLit "Constraint solver steps =")                               <+> int count <+> ppr context)  #endif diff --git a/compiler/typecheck/TcUnify.lhs b/compiler/typecheck/TcUnify.lhs index a6c9c478d3..572ad4437c 100644 --- a/compiler/typecheck/TcUnify.lhs +++ b/compiler/typecheck/TcUnify.lhs @@ -20,7 +20,7 @@ module TcUnify (    matchExpectedListTy, matchExpectedPArrTy,     matchExpectedTyConApp, matchExpectedAppTy,     matchExpectedFunTys, matchExpectedFunKind, -  wrapFunResCoercion +  wrapFunResCoercion, failWithMisMatch    ) where  #include "HsVersions.h" diff --git a/configure.ac b/configure.ac index d2deeb6c06..67d6b57f0f 100644 --- a/configure.ac +++ b/configure.ac @@ -265,6 +265,7 @@ checkVendor() {      esac  } +# Sync this with cTargetOS in compiler/ghc.mk  checkOS() {      case $1 in      linux|freebsd|netbsd|openbsd|dragonfly|osf1|osf3|hpux|linuxaout|kfreebsdgnu|freebsd2|solaris2|cygwin32|mingw32|darwin|gnu|nextstep2|nextstep3|sunos4|ultrix|irix|aix|haiku) @@ -631,8 +632,6 @@ FP_CHECK_DOCBOOK_DTD  FP_DOCBOOK_XSL  FP_PROG_DBLATEX -FP_PROG_HSTAGS -  dnl ** check for ghc-pkg command  FP_PROG_GHC_PKG diff --git a/docs/users_guide/flags.xml b/docs/users_guide/flags.xml index 73faae7f97..4a502b4b8c 100644 --- a/docs/users_guide/flags.xml +++ b/docs/users_guide/flags.xml @@ -898,6 +898,12 @@  	      <entry>dynamic</entry>  	      <entry><option>-XNoTransformListComp</option></entry>  	    </row> +        <row> +	      <entry><option>-XMonadComprehensions</option></entry> +	      <entry>Enable <link linkend="monad-comprehensions">monad comprehensions</link>.</entry> +	      <entry>dynamic</entry> +	      <entry><option>-XNoMonadComprehensions</option></entry> +	    </row>  	    <row>  	      <entry><option>-XUnliftedFFITypes</option></entry>  	      <entry>Enable unlifted FFI types.</entry> diff --git a/docs/users_guide/glasgow_exts.xml b/docs/users_guide/glasgow_exts.xml index 9ea3332463..89198c4264 100644 --- a/docs/users_guide/glasgow_exts.xml +++ b/docs/users_guide/glasgow_exts.xml @@ -1201,6 +1201,234 @@ output = [ x  </para>    </sect2> +   <!-- ===================== MONAD COMPREHENSIONS ===================== --> + +<sect2 id="monad-comprehensions"> +    <title>Monad comprehensions</title> +    <indexterm><primary>monad comprehensions</primary></indexterm> + +    <para> +        Monad comprehesions generalise the list comprehension notation, +        including parallel comprehensions  +        (<xref linkend="parallel-list-comprehensions"/>) and  +        transform comprenensions (<xref linkend="generalised-list-comprehensions"/>)  +        to work for any monad. +    </para> + +    <para>Monad comprehensions support:</para> + +    <itemizedlist> +        <listitem> +            <para> +                Bindings: +            </para> + +<programlisting> +[ x + y | x <- Just 1, y <- Just 2 ] +</programlisting> + +            <para> +                Bindings are translated with the <literal>(>>=)</literal> and +                <literal>return</literal> functions to the usual do-notation: +            </para> + +<programlisting> +do x <- Just 1 +   y <- Just 2 +   return (x+y) +</programlisting> + +        </listitem> +        <listitem> +            <para> +                Guards: +            </para> + +<programlisting> +[ x | x <- [1..10], x <= 5 ] +</programlisting> + +            <para> +                Guards are translated with the <literal>guard</literal> function, +                which requires a <literal>MonadPlus</literal> instance: +            </para> + +<programlisting> +do x <- [1..10] +   guard (x <= 5) +   return x +</programlisting> + +        </listitem> +        <listitem> +            <para> +                Transform statements (as with <literal>-XTransformListComp</literal>): +            </para> + +<programlisting> +[ x+y | x <- [1..10], y <- [1..x], then take 2 ] +</programlisting> + +            <para> +                This translates to: +            </para> + +<programlisting> +do (x,y) <- take 2 (do x <- [1..10] +                       y <- [1..x] +                       return (x,y)) +   return (x+y) +</programlisting> + +        </listitem> +        <listitem> +            <para> +                Group statements (as with <literal>-XTransformListComp</literal>): +            </para> + +<programlisting> +[ x | x <- [1,1,2,2,3], then group by x ] +[ x | x <- [1,1,2,2,3], then group by x using GHC.Exts.groupWith ] +[ x | x <- [1,1,2,2,3], then group using myGroup ] +</programlisting> + +            <para> +                The basic <literal>then group by e</literal> statement is +                translated using the <literal>mgroupWith</literal> function, which +                requires a <literal>MonadGroup</literal> instance, defined in +                <ulink url="&libraryBaseLocation;/Control-Monad-Group.html"><literal>Control.Monad.Group</literal></ulink>: +            </para> + +<programlisting> +do x <- mgroupWith (do x <- [1,1,2,2,3] +                       return x) +   return x +</programlisting> + +            <para> +                Note that the type of <literal>x</literal> is changed by the +                grouping statement. +            </para> + +            <para> +                The grouping function can also be defined with the +                <literal>using</literal> keyword. +            </para> + +        </listitem> +        <listitem> +            <para> +                Parallel statements (as with <literal>-XParallelListComp</literal>): +            </para> + +<programlisting> +[ (x+y) | x <- [1..10] +        | y <- [11..20] +        ] +</programlisting> + +            <para> +                Parallel statements are translated using the +                <literal>mzip</literal> function, which requires a +                <literal>MonadZip</literal> instance defined in +                <ulink url="&libraryBaseLocation;/Control-Monad-Zip.html"><literal>Control.Monad.Zip</literal></ulink>: +            </para> + +<programlisting> +do (x,y) <- mzip (do x <- [1..10] +                     return x) +                 (do y <- [11..20] +                     return y) +   return (x+y) +</programlisting> + +        </listitem> +    </itemizedlist> + +    <para> +        All these features are enabled by default if the +        <literal>MonadComprehensions</literal> extension is enabled. The types +        and more detailed examples on how to use comprehensions are explained +        in the previous chapters <xref +            linkend="generalised-list-comprehensions"/> and <xref +            linkend="parallel-list-comprehensions"/>. In general you just have +        to replace the type <literal>[a]</literal> with the type +        <literal>Monad m => m a</literal> for monad comprehensions. +    </para> + +    <para> +        Note: Even though most of these examples are using the list monad, +        monad comprehensions work for any monad. +        The <literal>base</literal> package offers all necessary instances for +        lists, which make <literal>MonadComprehensions</literal> backward +        compatible to built-in, transform and parallel list comprehensions. +    </para> +<para> More formally, the desugaring is as follows.  We write <literal>D[ e | Q]</literal> +to mean the desugaring of the monad comprehension <literal>[ e | Q]</literal>:  +<programlisting> +Expressions: e +Declarations: d +Lists of qualifiers: Q,R,S   + +-- Basic forms +D[ e | ]               = return e +D[ e | p <- e, Q ]     = e >>= \p -> D[ e | Q ] +D[ e | e, Q ]          = guard e >> \p -> D[ e | Q ] +D[ e | let d, Q ]      = let d in D[ e | Q ] + +-- Parallel comprehensions (iterate for multiple parallel branches) +D[ e | (Q | R), S ]    = mzip D[ Qv | Q ] D[ Rv | R ] >>= \(Qv,Rv) -> D[ e | S ] + +-- Transform comprehensions +D[ e | Q then f, R ]                  = f D[ Qv | Q ] >>= \Qv -> D[ e | R ] + +D[ e | Q then f by b, R ]             = f b D[ Qv | Q ] >>= \Qv -> D[ e | R ] + +D[ e | Q then group using f, R ]      = f D[ Qv | Q ] >>= \ys ->  +                                        case (fmap selQv1 ys, ..., fmap selQvn ys) of +                                 	     Qv -> D[ e | R ] + +D[ e | Q then group by b using f, R ] = f b D[ Qv | Q ] >>= \ys ->  +                                        case (fmap selQv1 ys, ..., fmap selQvn ys) of +                                           Qv -> D[ e | R ] + +where  Qv is the tuple of variables bound by Q (and used subsequently) +       selQvi is a selector mapping Qv to the ith component of Qv + +Operator     Standard binding       Expected type +-------------------------------------------------------------------- +return       GHC.Base               t1 -> m t2 +(>>=)        GHC.Base               m1 t1 -> (t2 -> m2 t3) -> m3 t3 +(>>)         GHC.Base               m1 t1 -> m2 t2         -> m3 t3 +guard        Control.Monad          t1 -> m t2 +fmap         GHC.Base               forall a b. (a->b) -> n a -> n b +mgroupWith   Control.Monad.Group    forall a. (a -> t) -> m1 a -> m2 (n a) +mzip         Control.Monad.Zip      forall a b. m a -> m b -> m (a,b) +</programlisting>                                           +The comprehension should typecheck when its desugaring would typecheck.  +</para> +<para> +Monad comprehensions support rebindable syntax (<xref linkend="rebindable-syntax"/>).   +Without rebindable +syntax, the operators from the "standard binding" module are used; with +rebindable syntax, the operators are looked up in the current lexical scope. +For example, parallel comprehensions will be typechecked and desugared +using whatever "<literal>mzip</literal>" is in scope. +</para> +<para> +The rebindable operators must have the "Expected type" given in the  +table above.  These types are surprisingly general.  For example, you can +use a bind operator with the type +<programlisting> +(>>=) :: T x y a -> (a -> T y z b) -> T x z b +</programlisting> +In the case of transform comprehensions, notice that the groups are +parameterised over some arbitrary type <literal>n</literal> (provided it +has an <literal>fmap</literal>, as well as +the comprehension being over an arbitrary monad. +</para> +</sect2> +     <!-- ===================== REBINDABLE SYNTAX ===================  -->  <sect2 id="rebindable-syntax"> diff --git a/ghc.spec.in b/ghc.spec.in index c8eab264c2..2a70043eea 100644 --- a/ghc.spec.in +++ b/ghc.spec.in @@ -177,7 +177,6 @@ fi  %{_prefix}/bin/ghci  %{_prefix}/bin/ghci-%{version}  %{_prefix}/bin/ghcprof -%{_prefix}/bin/hasktags  %{_prefix}/bin/hp2ps  %{_prefix}/bin/hpc  %{_prefix}/bin/hsc2hs-ghc diff --git a/includes/stg/SMP.h b/includes/stg/SMP.h index f1b0422009..52fd6f1bc6 100644 --- a/includes/stg/SMP.h +++ b/includes/stg/SMP.h @@ -306,6 +306,7 @@ load_load_barrier(void) {  #define store_load_barrier() /* nothing */  #define load_load_barrier()  /* nothing */ +#if !IN_STG_CODE || IN_STGCRUN  INLINE_HEADER StgWord  xchg(StgPtr p, StgWord w)  { @@ -337,6 +338,7 @@ atomic_dec(StgVolatilePtr p)  {      return --(*p);  } +#endif  #define VOLATILE_LOAD(p) ((StgWord)*((StgWord*)(p))) diff --git a/mk/build.mk.sample b/mk/build.mk.sample index a7764e2e46..216ca66c1b 100644 --- a/mk/build.mk.sample +++ b/mk/build.mk.sample @@ -136,15 +136,6 @@ endif  # -----------------------------------------------------------------------------  # Other settings that might be useful -# profiled RTS -#GhcRtsCcOpts =  -pg -g - -# Optimised/profiled RTS -#GhcRtsCcOpts = -O2 -pg - -#GhcRtsWithFrontPanel = YES -#SRC_HC_OPTS += `gtk-config --libs` -  # NoFib settings  NoFibWays =  STRIP_CMD = : diff --git a/mk/config.mk.in b/mk/config.mk.in index 8796ad4674..3749bce6b6 100644 --- a/mk/config.mk.in +++ b/mk/config.mk.in @@ -774,8 +774,6 @@ ALEX_VERSION		= @AlexVersion@  #  SRC_ALEX_OPTS		= -g -HSTAGS = @HstagsCmd@ -  # Should we build haddock docs?  HADDOCK_DOCS = YES  # And HsColour the sources? diff --git a/utils/Makefile b/utils/Makefile index 881d7d50b9..e522c32ba8 100644 --- a/utils/Makefile +++ b/utils/Makefile @@ -60,7 +60,7 @@ endif  WITH_BOOTSTRAPPING_COMPILER = installPackage ghc-pkg hsc2hs hpc -WITH_STAGE2 = installPackage ghc-pkg hasktags runghc hpc pwd haddock +WITH_STAGE2 = installPackage ghc-pkg runghc hpc pwd haddock  ifneq "$(NO_INSTALL_HSC2HS)" "YES"  WITH_STAGE2 += hsc2hs  endif | 
