Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make NuMatchingAny1Proof an alias for a quantified NuMatching constraint #8

Merged
merged 1 commit into from
Dec 5, 2022
Merged

Conversation

RyanGlScott
Copy link
Contributor

Previously, there was a global INCOHERENT instance of the form:

instance {-# INCOHERENT #-} NuMatchingAny1 f => NuMatching (f a) where
  nuMatchingProof = nuMatchingAny1Proof

This does not combine well with GHC 9.0+'s Template Haskell splice behavior, where instances written after a top-level declaration splice are not visible to instances written before the splice. In particular, this affects which instances GHC will consider when resolving instance overlap w.r.t. incoherent instances, so if you are not careful with the ordering of instances and TH splices, you can accidentally change the runtime behavior of your program. For more information, see GaloisInc/saw-script#1742 and https://gitlab.haskell.org/ghc/ghc/-/issues/22492.

This patch removes the INCOHERENT instance and instead makes NuMatchingAny1 an alias for a quantified constraint involving NuMatching:

class    (forall a. NuMatching (f a)) => NuMatchingAny1 (f :: k -> Type)
instance (forall a. NuMatching (f a)) => NuMatchingAny1 (f :: k -> Type)

(Alternatively, we could have made this a type synonym, but using a class has the advantage of not forcing downstream libraries to enable QuantifiedConstraints everywhere that NuMatchingAny1 is used.)

Aside from this change, the rest of this patch simply deletes NuMatchingAny1 instances, as the one instance above is the only instance we will ever need now.

Credit goes to Matthew Pickering for helping me identify this issue and for authoring a separate fix. I have tweaked his fix and turned it into this patch, adding him as a co-author in the process.

…straint

Previously, there was a global `INCOHERENT` instance of the form:

```hs
instance {-# INCOHERENT #-} NuMatchingAny1 f => NuMatching (f a) where
  nuMatchingProof = nuMatchingAny1Proof
```

This does not combine well with GHC 9.0+'s Template Haskell splice behavior,
where instances written after a top-level declaration splice are not visible to
instances written before the splice. In particular, this affects which instances
GHC will consider when resolving instance overlap w.r.t. incoherent instances,
so if you are not careful with the ordering of instances and TH splices, you
can accidentally change the _runtime_ behavior of your program. For more
information, see GaloisInc/saw-script#1742 and
https://gitlab.haskell.org/ghc/ghc/-/issues/22492.

This patch removes the `INCOHERENT` instance and instead makes `NuMatchingAny1`
an alias for a quantified constraint involving `NuMatching`:

```hs
class    (forall a. NuMatching (f a)) => NuMatchingAny1 (f :: k -> Type)
instance (forall a. NuMatching (f a)) => NuMatchingAny1 (f :: k -> Type)
```

(Alternatively, we could have made this a type synonym, but using a class has
the advantage of not forcing downstream libraries to enable
`QuantifiedConstraints` everywhere that `NuMatchingAny1` is used.)

Aside from this change, the rest of this patch simply deletes `NuMatchingAny1`
instances, as the one instance above is the only instance we will ever need now.

Credit goes to Matthew Pickering for helping me identify this issue and for
authoring a separate fix. I have tweaked his fix and turned it into this patch,
adding him as a co-author in the process.

Co-authored-by: Matthew Pickering <[email protected]>
RyanGlScott added a commit to GaloisInc/saw-script that referenced this pull request Dec 5, 2022
This makes the necessary changes to adapt to eddywestbrook/hobbits#8,
which turns `NuMatchingAny1` into an alias for a quantified `NuMatching`
constraint. See #1742 and https://gitlab.haskell.org/ghc/ghc/-/issues/22492 for
the motivation behind this.

One unfortunate hiccup with this patch is that combining quantified superclasses
and `TypeFamilies` doesn't quite work out of the box on pre-9.2 GHCs due to
https://gitlab.haskell.org/ghc/ghc/-/issues/14860. As a result, I have to
introduce explicit equality constraints to work around the issue. I have added
a `Note [QuantifiedConstraints + TypeFamilies trick]` to describe why the
workaround is necessary.

Credit goes to Matthew Pickering for helping me identify this issue and for
authoring a separate fix. I have tweaked his fix and turned it into this patch,
adding him as a co-author in the process.

Fixes #1742.

Co-authored-by: Matthew Pickering <[email protected]>
Copy link
Owner

@eddywestbrook eddywestbrook left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good, thanks Ryan!

@eddywestbrook eddywestbrook merged commit b88cbfc into eddywestbrook:master Dec 5, 2022
RyanGlScott added a commit to GaloisInc/saw-script that referenced this pull request Dec 5, 2022
This makes the necessary changes to adapt to eddywestbrook/hobbits#8,
which turns `NuMatchingAny1` into an alias for a quantified `NuMatching`
constraint. See #1742 and https://gitlab.haskell.org/ghc/ghc/-/issues/22492 for
the motivation behind this.

One unfortunate hiccup with this patch is that combining quantified superclasses
and `TypeFamilies` doesn't quite work out of the box on pre-9.2 GHCs due to
https://gitlab.haskell.org/ghc/ghc/-/issues/14860. As a result, I have to
introduce explicit equality constraints to work around the issue. I have added
a `Note [QuantifiedConstraints + TypeFamilies trick]` to describe why the
workaround is necessary.

Credit goes to Matthew Pickering for helping me identify this issue and for
authoring a separate fix. I have tweaked his fix and turned it into this patch,
adding him as a co-author in the process.

Fixes #1742.

Co-authored-by: Matthew Pickering <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants