Skip to content
Snippets Groups Projects

Example of delayed resolution of multiplicity constraints

Open Arnaud Spiwack requested to merge tweag/ghc:example-of-delayed-mult-constraint into master
1 unresolved thread

Closes #25130.

I've very much not satisfied with the word-smithing. To be honest, I'm not fully sure what's going on. @simonpj let me know what you think.

Merge request reports

Loading
Loading

Activity

Filter activity
  • Approvals
  • Assignees & reviewers
  • Comments (from bots)
  • Comments (from users)
  • Commits & branches
  • Edits
  • Labels
  • Lock status
  • Mentions
  • Merge request status
  • Tracking
1550 1550 How can we check that every call to `tcSubMult` returns `Refl`?
1551 1551 It might not be `Refl` *yet*.
1552 1552
1553 [TODO: add counterexample #25130]
1553 For instance
1554
1555 f :: T -> String
1556 f x = case x of
1557 MkT y -> ""
1558
1559 data T = forall a. MkT a
1560
1561 Here the case expression generates a unification variable p, for the
1562 multiplicity at which it consumes the scrutinee, and gives y multiplicity
1563 p. Since y isn't used, p is unified with Many. However, because of the
  • I don't understand this. Here f has a type signature f :: T %n-> String, where I have made explicit the multiplicity. But what is n? It's a complete type signature, so there is nothing unknown about it.

    Anyway x ends up in the envt with that multiplicity: x ::_n T, I think. Correct?

    OK now we get to the case expression. Maybe at a case expression we always make up a fresh multiplicity p to say how the scrutinee is used? (Is this described anywhere?) Let's suppose we do that.

    Now we have the pattern MkT y. Does that "see" p in any way? If so how? Maybe you are saying that we add y to the envt with type y ::_p a, where a is the existentially bound variable?

    "Since y isn't used, p is unified with Many". Where does that happen?

    This section is all about submultiplicty constraints being Refl. In this example, precisely which bit of code generates the submultiplicity-constraint-that-must-be-Ref?

    TL;DR: I think more detail would help. Perhaps a simpler example to start with? The simpler example can ignore the "might only be unified later" issue -- that topic can be the follow up ... "you might think that we can check for Refl right away, but this example shows..."

  • I don't understand this. Here f has a type signature f :: T %n-> String, where I have made explicit the multiplicity. But what is n? It's a complete type signature, so there is nothing unknown about it.

    The meaning T -> String is always T %Many -> String.

    Anyway x ends up in the envt with that multiplicity: x ::_n T, I think. Correct?

    Yes. Where n is Many

    OK now we get to the case expression. Maybe at a case expression we always make up a fresh multiplicity p to say how the scrutinee is used? (Is this described anywhere?) Let's suppose we do that.

    Indeed, this is what happens. I'm lacking a name for that (a lot of my documentation on linear types is plagued by my lack of creativity with naming stuff). The fact that case expressions have a multiplicity like this is described in the paper and the Linear Core document. The fact that this multiplicity is inferred is described in the proposal (we have not, on the other hand, even designed a syntax for making this multiplicity explicit :slight_frown: ).

    Now we have the pattern MkT y. Does that "see" p in any way? If so how? Maybe you are saying that we add y to the envt with type y ::_p a, where a is the existentially bound variable?

    Indeed. Specifically, the multiplicity of y in the environment is the multiplicity of the field (here 1), multiplied by the multiplicity of the case (here p).

    "Since y isn't used, p is unified with Many". Where does that happen?

    Are you looking for the precise function? It happens here: https://gitlab.haskell.org/ghc/ghc/-/blob/076c1a104f55750a49de03694786180bd78eb9b6/compiler/GHC/Tc/Gen/Pat.hs#L626-627 .

    This section is all about submultiplicty constraints being Refl. In this example, precisely which bit of code generates the submultiplicity-constraint-that-must-be-Ref?

    Every binder generates submultiplicity constraints, here it's the y binder which is our concern.

     TL;DR: I think more detail would help. Perhaps a simpler example to start with? The simpler example can ignore the "might only be unified later" issue -- that topic can be the follow up ... "you might think that we can check for Refl right away, but this example shows..."

    The goal of this example was specifically to answer this recurring question: why do we need to emit a constraint at all and can't solve submultiplicity constraints eagerly. Ok to move it a little out of the way if it feels distracting. I'm not too sure that this note needs another example, though. What an example of would it be?

  • why do we need to emit a constraint at all and can't solve submultiplicity constraints eagerly

    Exactly. So I need to see spot in the code where we say co <- unify m1 m2; emitMultiplicityCheck co, where it would not be OK to say zonk t1; zonk t2; compare them for equality.

    My point is that it'd be great to show an example where that is ok; but then show another where it isn't. At the moment, despite your explanation, I can't see the spot where that unification-that-should-be-refl takes place in GHC's source code, for the particular example you chose.

  • Please register or sign in to reply
Please register or sign in to reply
Loading