diff options
| author | Simon Peyton Jones <simonpj@microsoft.com> | 2012-03-04 08:18:09 +0000 | 
|---|---|---|
| committer | Simon Peyton Jones <simonpj@microsoft.com> | 2012-03-04 08:18:09 +0000 | 
| commit | 90de9736adada919b50a9a2ce5aae136f64c75fe (patch) | |
| tree | 5cd1ab8e08d8de896727008c0146e1bf1d2fec7e | |
| parent | eeba5437ee3c0f35a234bc76941d434438388ac3 (diff) | |
| download | haskell-90de9736adada919b50a9a2ce5aae136f64c75fe.tar.gz | |
Attach INLINE pagmas in mutually recursive bindings
This should fix #5895.  It seems that I was silently
ignoring INLINE pragmas in mutual recursion, which is
not the right thing at all.
| -rw-r--r-- | compiler/deSugar/DsBinds.lhs | 31 | ||||
| -rw-r--r-- | compiler/hsSyn/HsBinds.lhs | 8 | 
2 files changed, 33 insertions, 6 deletions
| diff --git a/compiler/deSugar/DsBinds.lhs b/compiler/deSugar/DsBinds.lhs index 09ab98fff1..4f94a1c3e9 100644 --- a/compiler/deSugar/DsBinds.lhs +++ b/compiler/deSugar/DsBinds.lhs @@ -153,8 +153,10 @@ dsHsBind (AbsBinds { abs_tvs = tyvars, abs_ev_vars = dicts  dsHsBind (AbsBinds { abs_tvs = tyvars, abs_ev_vars = dicts                     , abs_exports = exports, abs_ev_binds = ev_binds                     , abs_binds = binds }) +         -- See Note [Desugaring AbsBinds]    = do  { bind_prs    <- ds_lhs_binds binds -        ; let core_bind = Rec (fromOL bind_prs) +        ; let core_bind = Rec [ makeCorePair (add_inline lcl_id) False 0 rhs +                              | (lcl_id, rhs) <- fromOL bind_prs ]  	      	-- Monomorphic recursion possible, hence Rec  	      tup_expr     = mkBigCoreVarTup locals @@ -176,13 +178,28 @@ dsHsBind (AbsBinds { abs_tvs = tyvars, abs_ev_vars = dicts  			         mkVarApps (Var poly_tup_id) (tyvars ++ dicts)                             rhs_for_spec = Let (NonRec poly_tup_id poly_tup_rhs) rhs  		     ; (spec_binds, rules) <- dsSpecs rhs_for_spec spec_prags -		     ; let global' = addIdSpecialisations global rules +		     ; let global' = (global `setInlinePragma` defaultInlinePragma) +                                             `addIdSpecialisations` rules +                           -- Kill the INLINE pragma because it applies to +                           -- the user written (local) function.  The global +                           -- Id is just the selector.  Hmm.    		     ; return ((global', rhs) `consOL` spec_binds) }          ; export_binds_s <- mapM mk_bind exports  	; return ((poly_tup_id, poly_tup_rhs) `consOL`   		    concatOL export_binds_s) } +  where +    inline_env :: IdEnv Id   -- Maps a monomorphic local Id to one with +                             -- the inline pragma from the source +                             -- The type checker put the inline pragma +                             -- on the *global* Id, so we need to transfer it +    inline_env = mkVarEnv [ (lcl_id, setInlinePragma lcl_id prag) +                          | ABE { abe_mono = lcl_id, abe_poly = gbl_id } <- exports +                          , let prag = idInlinePragma gbl_id ] + +    add_inline :: Id -> Id    -- tran +    add_inline lcl_id = lookupVarEnv inline_env lcl_id `orElse` lcl_id  ------------------------  makeCorePair :: Id -> Bool -> Arity -> CoreExpr -> (Id, CoreExpr) @@ -219,6 +236,16 @@ dictArity :: [Var] -> Arity  dictArity dicts = count isId dicts  \end{code} +[Desugaring AbsBinds] +~~~~~~~~~~~~~~~~~~~~~ +In the general AbsBinds case we desugar the binding to this: + +       tup a (d:Num a) = let fm = ...gm... +                             gm = ...fm... +                         in (fm,gm) +       f a d = case tup a d of { (fm,gm) -> fm } +       g a d = case tup a d of { (fm,gm) -> fm } +  Note [Rules and inlining]  ~~~~~~~~~~~~~~~~~~~~~~~~~  Common special case: no type or dictionary abstraction diff --git a/compiler/hsSyn/HsBinds.lhs b/compiler/hsSyn/HsBinds.lhs index bb8b337a00..f756578e2d 100644 --- a/compiler/hsSyn/HsBinds.lhs +++ b/compiler/hsSyn/HsBinds.lhs @@ -175,12 +175,12 @@ data HsBindLR idL idR          --  of this last construct.)  data ABExport id -  = ABE { abe_poly  :: id +  = ABE { abe_poly  :: id           -- Any INLINE pragmas is attached to this Id          , abe_mono  :: id -        , abe_wrap  :: HsWrapper  -- See Note [AbsBinds wrappers] +        , abe_wrap  :: HsWrapper    -- See Note [AbsBinds wrappers]               -- Shape: (forall abs_tvs. abs_ev_vars => abe_mono) ~ abe_poly -        , abe_prags :: TcSpecPrags } -  deriving (Data, Typeable) +        , abe_prags :: TcSpecPrags  -- SPECIALISE pragmas +  } deriving (Data, Typeable)  placeHolderNames :: NameSet  -- Used for the NameSet in FunBind and PatBind prior to the renamer | 
