[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