[Git][ghc/ghc][wip/T24359] Comments
Simon Peyton Jones (@simonpj)
gitlab at gitlab.haskell.org
Thu Mar 13 09:51:08 UTC 2025
Simon Peyton Jones pushed to branch wip/T24359 at Glasgow Haskell Compiler / GHC
Commits:
b8323c5e by Simon Peyton Jones at 2025-03-13T09:50:30+00:00
Comments
Finished Sig.hs, but some questions in Binds.hs
- - - - -
2 changed files:
- compiler/GHC/HsToCore/Binds.hs
- compiler/GHC/Tc/Gen/Sig.hs
Changes:
=====================================
compiler/GHC/HsToCore/Binds.hs
=====================================
@@ -789,20 +789,23 @@ The restrictions are:
4. Unlifted binds may not be recursive. Checked in second clause of ds_val_bind.
-Note [Desugaring SPECIALISE pragmas]
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+Note [Desugaring new-form SPECIALISE pragmas]
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+"New-form" SPECIALISE pragmas generate a SpecPragE record in the typechecker,
+which is desugared in this module by `dsSpec`. For the context see
+Note [Handling new-form SPECIALISE pragmas] in GHC.Tc.Gen.Sig
+
Suppose we have f :: forall p q. (Ord p, Eq q) => p -> q -> q, and a pragma
{-# SPECIALISE forall x. f @[a] @[Int] x [3,4] #-}
-In `dsSpec` on `SpecPragE`, the SPECIALISE pragma has an expression `the_call`
-that desugars to something like
-
- forall @a (d:Ord a) (x:[a]).
- let d2:Ord [a] = $dfOrdList d
- d3:Eq [Int] = $dfEqList $dfEqInt
- in f @[a] @[Int] d2 d3 x [3,4]
+In `dsSpec` the `SpecPragE` will look something like this:
+ SpecPragE { spe_fn_id = f
+ , spe_bndrs = @a (d:Ord a) (x:[a])
+ , spe_call = let d2:Ord [a] = $dfOrdList d
+ d3:Eq [Int] = $dfEqList $dfEqInt
+ in f @[a] @[Int] d2 d3 x [3,4] }
We want to get
RULE forall a (d2:Ord a) (d3:Eq [Int]) (x:[a]).
@@ -814,28 +817,37 @@ We want to get
in <f-rhs> @[a] @[Int] d2 d3 x [3,4]
Notice that
-(SP1) If the expression had a type signature, such as
+
+(SP1) If the expression in the SPECIALISE pragma had a type signature, such as
SPECIALISE f :: Eq b => Int -> b -> b
then the desugared expression may have type abstractions and applications
"in the way", like this:
(/\b. (\d:Eq b). let d1 = $dfOrdInt in f @Int @b d1 d) @b (d2:Eq b)
+ The lambdas come from the type signature, which is then re-instantiated,
+ hence the applications of those lambdas.
+
We use the simple optimiser to simplify this to
let { d = d2; d1 = $dfOrdInt } in f @Int @b (d2:Eq b)
- Do no inlining in this "simple optimiser" phase: use `simpleOptExprNoInline`.
- E.g. we don't want to turn
+
+ Important: do no inlining in this "simple optimiser" phase:
+ use `simpleOptExprNoInline`. E.g. we don't want to turn
let { d1=d; d2=d } in f d1 d2 --> f d d
because the latter is harder to match.
(SP2) the function `prepareSpecLHS` takes the simplified LHS `core_call` and
splits those dictionary bindings into two:
- * Bindings like
+ (SP2a) Bindings like
d3:Eq [Int] = $dfEqList $dfEqInt
depend only on constants and move to the specialised fuction body. That is
crucial -- it makes those specialised methods available in the specialised
body. This are the `spec_const_binds`.
- * Bindings like
+ (SP2b) Bindings like
+ d1 = d
+ Suprisingly, we want to dis
+
+ (SP2c) Bindings like
d2:Ord [a] = $dfOrdList d
depend on a locally-quantifed evidence variable `d`.
Surprisingly, /we want to drop these bindings entirely!/
@@ -848,6 +860,11 @@ Notice that
particular, we can't reliably get a (d:Ord a) dictionary from the supplied
(d2:Eq [a]) argument.
+ Note: because of Note [Fully solving constraints for specialisation] in
+ GHC.Tc.Gen.Sig, there won't /be/ any such bindings -- they aren't
+ fully solved. But it earlier iterations there were, and it does no
+ harm to handle them.
+
Finally, inside those dictionary bindings we should find the call of the
function itself
f @[a] @[Int] d2 d3 x [3,4]
=====================================
compiler/GHC/Tc/Gen/Sig.hs
=====================================
@@ -765,12 +765,15 @@ This is done in three parts.
each original Wanted is either fully solved or left untouched.
See Note [Fully solving constraints for specialisation].
- (3) Compute the constraints to quantify over, using `getRuleQuantCts`.
+ (3) Compute the constraints to quantify over, using `getRuleQuantCts` on
+ the unsolved constraints returned by (2).
- (4) Emit the residual (non-quantified) constraints, and wrap the
+ (4) Emit the residual (non-solved, non-quantified) constraints, and wrap the
expression in a let binding for those constraints.
- (5) Store all the information in a 'SpecPragE' record, to be consumed
+ (5) Wrap the call in the combined evidence bindings from steps (2) and (4)
+
+ (6) Store all the information in a 'SpecPragE' record, to be consumed
by the desugarer.
B. Zonker: `GHC.Tc.Zonk.Type.zonkLTcSpecPrags`
@@ -786,7 +789,7 @@ This is done in three parts.
(1) Simplify the expression. This is important because a type signature in
the expression will have led to type/dictionary abstractions/applications.
- Now it should look like
+ After simplification it should look like
let <dict-binds> in f d1 d2 d3
(2) `prepareSpecLHS` identifies the `spec_const_binds`, discards the other
@@ -799,23 +802,11 @@ This is done in three parts.
Note [Fully solving constraints for specialisation]
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
As far as specialisation is concerned, it is actively harmful to simplify
-constraints without fully solving them. Two key examples:
-
-f :: ∀ a t. (Eq a, ∀x. Eq x => Eq (t x)). t a -> Char
-{-# SPECIALISE f @Int #-}
-
- Typechecking 'f' will result in [W] Eq Int, [W] ∀x. Eq x => Eq (t x).
- We absolutely MUST leave the quantified constraint alone, because we want to
- quantify over it. If we were to try to simplify it, we would emit an
- implication and would thereafter never be able to quantify over the original
- quantified constraint.
+constraints without /fully/ solving them. Two key examples:
- However, we still need to simplify quantified constraints that can be fully
- solved from instances, otherwise we would never be able to specialise them
- away. Example: {-# SPECIALISE f @a @[] #-}.
-
-g :: ∀ a. Eq a => a -> Bool
-{-# SPECIALISE g @[e] #-}
+* Type-class instances
+ g :: ∀ a. Eq a => a -> Bool
+ {-# SPECIALISE g @[e] #-}
Typechecking 'g' will result in [W] Eq [e]. Were we to simplify this to
[W] Eq e, we would have difficulty generating a RULE for the specialisation:
@@ -832,10 +823,25 @@ g :: ∀ a. Eq a => a -> Bool
can't do anything useful from the knowledge that a dictionary for 'Eq [e]' is
constructed from a dictionary for 'Eq e' using the 'Eq' instance for lists.
- Note however that it is less important to tackle this problem in the typechecker,
- as the desugarer would still be able to generate the correct RULE if we did
- simplify 'Eq [e]' to 'Eq e'. See the second bullet point in (SP2) in
- Note [Desugaring SPECIALISE pragmas] in GHC.HsToCore.Binds.
+ We could in principle tackle this problem in the desugarer, by discarding the
+ evidence for (d :: Eq [e]), and quantifiying over it instead: see the second
+ bullet point in (SP2) in Note [Desugaring SPECIALISE pragmas] in GHC.HsToCore.Binds.
+
+ BUT that doesn't work for quantified constraints, as we see next.
+
+* Quantified constraints
+ f :: ∀ a t. (Eq a, ∀x. Eq x => Eq (t x)). t a -> Char
+ {-# SPECIALISE f @Int #-}
+
+ Typechecking 'f' will result in [W] Eq Int, [W] ∀x. Eq x => Eq (t x).
+ We absolutely MUST leave the quantified constraint alone, because we want to
+ quantify over it. If we were to try to simplify it, we would emit an
+ implication and would thereafter never be able to quantify over the original
+ quantified constraint.
+
+ However, we still need to simplify quantified constraints that can be
+ /fully solved/ from instances, otherwise we would never be able to
+ specialise them away. Example: {-# SPECIALISE f @a @[] #-}.
The conclusion is this:
@@ -1030,15 +1036,22 @@ tcSpecPrag poly_id (SpecSigE nm rule_bndrs spec_e inl)
solveWanteds spec_e_wanted
; spec_e_wanted <- liftZonkM $ zonkWC spec_e_wanted
- -- (3) Compute which constraints to quantify over.
+ -- (3) Compute which constraints to quantify over, by looking
+ -- at the unsolved constraints from (2)
; (quant_cands, residual_wc) <- getRuleQuantCts spec_e_wanted
- -- (4) Emit the residual constraints (that we are not quantifying over)
+ -- (4) Emit the residual constraints (i.e. ones that we have
+ -- not solved in (2) nor quantified in (3)
+ -- NB: use the same `ev_binds_var` as (2), so the bindings
+ -- for (2) and (4) are combined
; let tv_bndrs = filter isTyVar rule_bndrs'
qevs = map ctEvId (bagToList quant_cands)
; emitResidualConstraints rhs_tclvl skol_info_anon ev_binds_var
emptyVarSet tv_bndrs qevs
residual_wc
+
+ -- (5) Wrap the call in the combined evidence bindings
+ -- from steps (2) and (4)
; let lhs_call = mkLHsWrap (WpLet (TcEvBinds ev_binds_var)) tc_spec_e
; ev_binds <- getTcEvBindsMap ev_binds_var
@@ -1057,7 +1070,7 @@ tcSpecPrag poly_id (SpecSigE nm rule_bndrs spec_e inl)
, text "ev_binds:" <+> ppr ev_binds
]
- -- (5) Store the results in a SpecPragE record, which will be
+ -- (6) Store the results in a SpecPragE record, which will be
-- zonked and then consumed by the desugarer.
; return [SpecPragE { spe_fn_nm = nm
View it on GitLab: https://gitlab.haskell.org/ghc/ghc/-/commit/b8323c5e05bab4712038bdabe8e1eb68ed35f6bf
--
View it on GitLab: https://gitlab.haskell.org/ghc/ghc/-/commit/b8323c5e05bab4712038bdabe8e1eb68ed35f6bf
You're receiving this email because of your account on gitlab.haskell.org.
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://mail.haskell.org/pipermail/ghc-commits/attachments/20250313/69ae107d/attachment-0001.html>
More information about the ghc-commits
mailing list