summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSimon Peyton Jones <simonpj@microsoft.com>2011-11-16 10:36:50 +0000
committerSimon Peyton Jones <simonpj@microsoft.com>2011-11-16 10:36:50 +0000
commit04de986e978d89eb841b113e9f5b0c2a8f0cc101 (patch)
treef0210e3aa77b39b01324ad5a51cf2372e1166572
parent40a3361f5e8311505982a599fa74f44fc6f7bcaf (diff)
downloadhaskell-04de986e978d89eb841b113e9f5b0c2a8f0cc101.tar.gz
Further wibbles to calcUnfoldingGuidance, with documentation of same
I'd gotten into a state in which top-level x = y bindings weren't getting inlined!
-rw-r--r--compiler/coreSyn/CoreUnfold.lhs82
1 files changed, 44 insertions, 38 deletions
diff --git a/compiler/coreSyn/CoreUnfold.lhs b/compiler/coreSyn/CoreUnfold.lhs
index 8cbf4acea2..930041dea4 100644
--- a/compiler/coreSyn/CoreUnfold.lhs
+++ b/compiler/coreSyn/CoreUnfold.lhs
@@ -174,8 +174,7 @@ mkUnfolding src top_lvl is_bottoming expr
uf_guidance = guidance }
where
is_cheap = exprIsCheap expr
- (arity, guidance) = calcUnfoldingGuidance is_cheap
- opt_UF_CreationThreshold expr
+ (arity, guidance) = calcUnfoldingGuidance expr
-- Sometimes during simplification, there's a large let-bound thing
-- which has been substituted, and so is now dead; so 'expr' contains
-- two copies of the thing while the occurrence-analysed expression doesn't
@@ -217,14 +216,13 @@ inlineBoringOk e
go _ _ = boringCxtNotOk
calcUnfoldingGuidance
- :: Bool -- True <=> the rhs is cheap, or we want to treat it
- -- as cheap (INLINE things)
- -> Int -- Bomb out if size gets bigger than this
- -> CoreExpr -- Expression to look at
+ :: CoreExpr -- Expression to look at
-> (Arity, UnfoldingGuidance)
-calcUnfoldingGuidance expr_is_cheap bOMB_OUT_SIZE expr
+calcUnfoldingGuidance expr
= case collectBinders expr of { (bndrs, body) ->
let
+ bOMB_OUT_SIZE = opt_UF_CreationThreshold
+ -- Bomb out if size gets bigger than this
val_bndrs = filter isId bndrs
n_val_bndrs = length val_bndrs
@@ -232,8 +230,7 @@ calcUnfoldingGuidance expr_is_cheap bOMB_OUT_SIZE expr
= case (sizeExpr (iUnbox bOMB_OUT_SIZE) val_bndrs body) of
TooBig -> UnfNever
SizeIs size cased_bndrs scrut_discount
- | uncondInline n_val_bndrs (iBox size)
- , expr_is_cheap
+ | uncondInline expr n_val_bndrs (iBox size)
-> UnfWhen unSaturatedOk boringCxtOk -- Note [INLINE for small functions]
| otherwise
-> UnfIfGoodArgs { ug_args = map (discount cased_bndrs) val_bndrs
@@ -290,7 +287,6 @@ and similar friends. See Note [Bottoming floats] in SetLevels.
Do not re-inline them! But we *do* still inline if they are very small
(the uncondInline stuff).
-
Note [INLINE for small functions]
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Consider {-# INLINE f #-}
@@ -303,44 +299,54 @@ inline unconditionally, regardless of how boring the context is.
Things to note:
- * We inline *unconditionally* if inlined thing is smaller (using sizeExpr)
- than the thing it's replacing. Notice that
+(1) We inline *unconditionally* if inlined thing is smaller (using sizeExpr)
+ than the thing it's replacing. Notice that
(f x) --> (g 3) -- YES, unconditionally
(f x) --> x : [] -- YES, *even though* there are two
-- arguments to the cons
x --> g 3 -- NO
x --> Just v -- NO
- It's very important not to unconditionally replace a variable by
- a non-atomic term.
-
-* We do this even if the thing isn't saturated, else we end up with the
- silly situation that
- f x y = x
- ...map (f 3)...
- doesn't inline. Even in a boring context, inlining without being
- saturated will give a lambda instead of a PAP, and will be more
- efficient at runtime.
-
-* However, when the function's arity > 0, we do insist that it
- has at least one value argument at the call site. Otherwise we find this:
- f = /\a \x:a. x
- d = /\b. MkD (f b)
- If we inline f here we get
- d = /\b. MkD (\x:b. x)
- and then prepareRhs floats out the argument, abstracting the type
- variables, so we end up with the original again!
-
+ It's very important not to unconditionally replace a variable by
+ a non-atomic term.
+
+(2) We do this even if the thing isn't saturated, else we end up with the
+ silly situation that
+ f x y = x
+ ...map (f 3)...
+ doesn't inline. Even in a boring context, inlining without being
+ saturated will give a lambda instead of a PAP, and will be more
+ efficient at runtime.
+
+(3) However, when the function's arity > 0, we do insist that it
+ has at least one value argument at the call site. (This check is
+ made in the UnfWhen case of callSiteInline.) Otherwise we find this:
+ f = /\a \x:a. x
+ d = /\b. MkD (f b)
+ If we inline f here we get
+ d = /\b. MkD (\x:b. x)
+ and then prepareRhs floats out the argument, abstracting the type
+ variables, so we end up with the original again!
+
+(4) We must be much more cautious about arity-zero things. Consider
+ let x = y +# z in ...
+ In *size* terms primops look very small, because the generate a
+ single instruction, but we do not want to unconditionally replace
+ every occurrence of x with (y +# z). So we only do the
+ unconditional-inline thing for *trivial* expressions.
+
+ NB: you might think that PostInlineUnconditionally would do this
+ but it doesn't fire for top-level things; see SimplUtils
+ Note [Top level and postInlineUnconditionally]
\begin{code}
-uncondInline :: Arity -> Int -> Bool
+uncondInline :: CoreExpr -> Arity -> Int -> Bool
-- Inline unconditionally if there no size increase
-- Size of call is arity (+1 for the function)
-- See Note [INLINE for small functions]
-uncondInline arity size
- | arity == 0 = False -- Never unconditionally inline non-lambda
- -- PostInlineUnconditionally will do that
- | otherwise = size <= 10 * (arity + 1)
+uncondInline rhs arity size
+ | arity > 0 = size <= 10 * (arity + 1) -- See Note [INLINE for small functions] (1)
+ | otherwise = exprIsTrivial rhs -- See Note [INLINE for small functions] (4)
\end{code}
@@ -907,7 +913,7 @@ tryUnfolding dflags id lone_variable
UnfWhen unsat_ok boring_ok
-> (enough_args && (boring_ok || some_benefit), empty )
- where -- See Note [INLINE for small functions]
+ where -- See Note [INLINE for small functions (3)]
enough_args = saturated || (unsat_ok && n_val_args > 0)
UnfIfGoodArgs { ug_args = arg_discounts, ug_res = res_discount, ug_size = size }