summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKrzysztof Gogolewski <krzysztof.gogolewski@tweag.io>2021-08-06 19:35:24 +0200
committerMarge Bot <ben+marge-bot@smart-cactus.org>2021-09-17 09:35:53 -0400
commit4564f00fdeb5e072e8f91fec72a6393f0e3f0703 (patch)
tree9cbbc1791aac4c6cf07c08abf1c1c0f05f457e96
parent885f17c89919d21815365da71eb7f9c489e5bfa3 (diff)
downloadhaskell-4564f00fdeb5e072e8f91fec72a6393f0e3f0703.tar.gz
Improve pretty-printer defaulting logic (#19361)
When determining whether to default a RuntimeRep or Multiplicity variable, use isMetaTyVar to distinguish between metavariables (which can be hidden) and skolems (which cannot).
-rw-r--r--compiler/GHC/Iface/Type.hs79
-rw-r--r--compiler/GHC/Tc/Utils/TcType.hs-boot4
-rw-r--r--compiler/GHC/Types/Var.hs-boot1
-rw-r--r--testsuite/tests/linear/should_fail/T19361.hs9
-rw-r--r--testsuite/tests/linear/should_fail/T19361.stderr10
-rw-r--r--testsuite/tests/linear/should_fail/all.T1
-rw-r--r--testsuite/tests/typecheck/should_fail/T15883b.stderr9
-rw-r--r--testsuite/tests/typecheck/should_fail/T15883c.stderr9
-rw-r--r--testsuite/tests/typecheck/should_fail/T15883d.stderr10
-rw-r--r--testsuite/tests/typecheck/should_fail/T15883e.stderr10
10 files changed, 92 insertions, 50 deletions
diff --git a/compiler/GHC/Iface/Type.hs b/compiler/GHC/Iface/Type.hs
index 58410467d3..422091784a 100644
--- a/compiler/GHC/Iface/Type.hs
+++ b/compiler/GHC/Iface/Type.hs
@@ -87,6 +87,7 @@ import GHC.Utils.Outputable
import GHC.Data.FastString
import GHC.Utils.Misc
import GHC.Utils.Panic
+import {-# SOURCE #-} GHC.Tc.Utils.TcType ( isMetaTyVar, isTyConableTyVar )
import Data.Maybe( isJust )
import qualified Data.Semigroup as Semi
@@ -1006,18 +1007,25 @@ This is done in a pass right before pretty-printing
This applies to /quantified/ variables like 'w' above. What about
variables that are /free/ in the type being printed, which certainly
-happens in error messages. Suppose (#16074) we are reporting a
-mismatch between two skolems
+happens in error messages. Suppose (#16074, #19361) we are reporting a
+mismatch between skolems
(a :: RuntimeRep) ~ (b :: RuntimeRep)
-We certainly don't want to say "Can't match LiftedRep ~ LiftedRep"!
+ or
+ (m :: Multiplicity) ~ Many
+We certainly don't want to say "Can't match LiftedRep with LiftedRep" or
+"Can't match Many with Many"!
But if we are printing the type
- (forall (a :: TYPE r). blah
+ (forall (a :: TYPE r). blah)
we do want to turn that (free) r into LiftedRep, so it prints as
(forall a. blah)
-Conclusion: keep track of whether we are in the kind of a
-binder; only if so, convert free RuntimeRep variables to LiftedRep.
+We use isMetaTyVar to distinguish between those two situations:
+metavariables are converted, skolem variables are not.
+
+There's one exception though: TyVarTv metavariables should not be defaulted,
+as they appear during kind-checking of "newtype T :: TYPE r where..."
+(test T18357a). Therefore, we additionally test for isTyConableTyVar.
-}
-- | Default 'RuntimeRep' variables to 'LiftedRep', and 'Multiplicity'
@@ -1039,65 +1047,68 @@ binder; only if so, convert free RuntimeRep variables to LiftedRep.
-- type signatures (e.g. ($)). See Note [Defaulting RuntimeRep variables]
-- and #11549 for further discussion.
defaultNonStandardVars :: Bool -> Bool -> IfaceType -> IfaceType
-defaultNonStandardVars do_runtimereps do_multiplicities ty = go False emptyFsEnv ty
+defaultNonStandardVars do_runtimereps do_multiplicities ty = go emptyFsEnv ty
where
- go :: Bool -- True <=> Inside the kind of a binder
- -> FastStringEnv IfaceType -- Set of enclosing forall-ed RuntimeRep/Multiplicity variables
+ go :: FastStringEnv IfaceType -- Set of enclosing forall-ed RuntimeRep/Multiplicity variables
-> IfaceType
-> IfaceType
- go ink subs (IfaceForAllTy (Bndr (IfaceTvBndr (var, var_kind)) argf) ty)
+ go subs (IfaceForAllTy (Bndr (IfaceTvBndr (var, var_kind)) argf) ty)
| isInvisibleArgFlag argf -- Don't default *visible* quantification
-- or we get the mess in #13963
, Just substituted_ty <- check_substitution var_kind
= let subs' = extendFsEnv subs var substituted_ty
-- Record that we should replace it with LiftedRep,
-- and recurse, discarding the forall
- in go ink subs' ty
+ in go subs' ty
- go ink subs (IfaceForAllTy bndr ty)
- = IfaceForAllTy (go_ifacebndr subs bndr) (go ink subs ty)
+ go subs (IfaceForAllTy bndr ty)
+ = IfaceForAllTy (go_ifacebndr subs bndr) (go subs ty)
- go _ subs ty@(IfaceTyVar tv) = case lookupFsEnv subs tv of
+ go subs ty@(IfaceTyVar tv) = case lookupFsEnv subs tv of
Just s -> s
Nothing -> ty
- go in_kind _ ty@(IfaceFreeTyVar tv)
+ go _ ty@(IfaceFreeTyVar tv)
-- See Note [Defaulting RuntimeRep variables], about free vars
- | in_kind && do_runtimereps && GHC.Core.Type.isRuntimeRepTy (tyVarKind tv)
+ | do_runtimereps && GHC.Core.Type.isRuntimeRepTy (tyVarKind tv)
+ , isMetaTyVar tv
+ , isTyConableTyVar tv
= liftedRep_ty
| do_multiplicities && GHC.Core.Type.isMultiplicityTy (tyVarKind tv)
+ , isMetaTyVar tv
+ , isTyConableTyVar tv
= many_ty
| otherwise
= ty
- go ink subs (IfaceTyConApp tc tc_args)
- = IfaceTyConApp tc (go_args ink subs tc_args)
+ go subs (IfaceTyConApp tc tc_args)
+ = IfaceTyConApp tc (go_args subs tc_args)
- go ink subs (IfaceTupleTy sort is_prom tc_args)
- = IfaceTupleTy sort is_prom (go_args ink subs tc_args)
+ go subs (IfaceTupleTy sort is_prom tc_args)
+ = IfaceTupleTy sort is_prom (go_args subs tc_args)
- go ink subs (IfaceFunTy af w arg res)
- = IfaceFunTy af (go ink subs w) (go ink subs arg) (go ink subs res)
+ go subs (IfaceFunTy af w arg res)
+ = IfaceFunTy af (go subs w) (go subs arg) (go subs res)
- go ink subs (IfaceAppTy t ts)
- = IfaceAppTy (go ink subs t) (go_args ink subs ts)
+ go subs (IfaceAppTy t ts)
+ = IfaceAppTy (go subs t) (go_args subs ts)
- go ink subs (IfaceCastTy x co)
- = IfaceCastTy (go ink subs x) co
+ go subs (IfaceCastTy x co)
+ = IfaceCastTy (go subs x) co
- go _ _ ty@(IfaceLitTy {}) = ty
- go _ _ ty@(IfaceCoercionTy {}) = ty
+ go _ ty@(IfaceLitTy {}) = ty
+ go _ ty@(IfaceCoercionTy {}) = ty
go_ifacebndr :: FastStringEnv IfaceType -> IfaceForAllBndr -> IfaceForAllBndr
go_ifacebndr subs (Bndr (IfaceIdBndr (w, n, t)) argf)
- = Bndr (IfaceIdBndr (w, n, go True subs t)) argf
+ = Bndr (IfaceIdBndr (w, n, go subs t)) argf
go_ifacebndr subs (Bndr (IfaceTvBndr (n, t)) argf)
- = Bndr (IfaceTvBndr (n, go True subs t)) argf
+ = Bndr (IfaceTvBndr (n, go subs t)) argf
- go_args :: Bool -> FastStringEnv IfaceType -> IfaceAppArgs -> IfaceAppArgs
- go_args _ _ IA_Nil = IA_Nil
- go_args ink subs (IA_Arg ty argf args)
- = IA_Arg (go ink subs ty) argf (go_args ink subs args)
+ go_args :: FastStringEnv IfaceType -> IfaceAppArgs -> IfaceAppArgs
+ go_args _ IA_Nil = IA_Nil
+ go_args subs (IA_Arg ty argf args)
+ = IA_Arg (go subs ty) argf (go_args subs args)
check_substitution :: IfaceType -> Maybe IfaceType
check_substitution (IfaceTyConApp tc _)
diff --git a/compiler/GHC/Tc/Utils/TcType.hs-boot b/compiler/GHC/Tc/Utils/TcType.hs-boot
index dc5f4cf73f..6b808dd7ab 100644
--- a/compiler/GHC/Tc/Utils/TcType.hs-boot
+++ b/compiler/GHC/Tc/Utils/TcType.hs-boot
@@ -1,8 +1,12 @@
module GHC.Tc.Utils.TcType where
import GHC.Utils.Outputable( SDoc )
+import GHC.Prelude ( Bool )
+import {-# SOURCE #-} GHC.Types.Var ( TcTyVar )
data MetaDetails
data TcTyVarDetails
pprTcTyVarDetails :: TcTyVarDetails -> SDoc
vanillaSkolemTv :: TcTyVarDetails
+isMetaTyVar :: TcTyVar -> Bool
+isTyConableTyVar :: TcTyVar -> Bool
diff --git a/compiler/GHC/Types/Var.hs-boot b/compiler/GHC/Types/Var.hs-boot
index f96157540a..1882a86d33 100644
--- a/compiler/GHC/Types/Var.hs-boot
+++ b/compiler/GHC/Types/Var.hs-boot
@@ -18,4 +18,5 @@ data Specificity
type TyVar = Var
type Id = Var
type TyCoVar = Id
+type TcTyVar = Var
type InvisTVBinder = VarBndr TyVar Specificity
diff --git a/testsuite/tests/linear/should_fail/T19361.hs b/testsuite/tests/linear/should_fail/T19361.hs
new file mode 100644
index 0000000000..503b299a0b
--- /dev/null
+++ b/testsuite/tests/linear/should_fail/T19361.hs
@@ -0,0 +1,9 @@
+{-# LANGUAGE LinearTypes #-}
+
+module T19361 where
+
+f :: a %m -> a
+f x = g x
+
+g :: a -> a
+g x = x
diff --git a/testsuite/tests/linear/should_fail/T19361.stderr b/testsuite/tests/linear/should_fail/T19361.stderr
new file mode 100644
index 0000000000..1d7746786a
--- /dev/null
+++ b/testsuite/tests/linear/should_fail/T19361.stderr
@@ -0,0 +1,10 @@
+
+T19361.hs:6:3: error:
+ • Couldn't match type ‘m’ with ‘'Many’
+ arising from multiplicity of ‘x’
+ ‘m’ is a rigid type variable bound by
+ the type signature for:
+ f :: forall a. a -> a
+ at T19361.hs:5:1-14
+ • In an equation for ‘f’: f x = g x
+ • Relevant bindings include f :: a %m -> a (bound at T19361.hs:6:1)
diff --git a/testsuite/tests/linear/should_fail/all.T b/testsuite/tests/linear/should_fail/all.T
index 89363cba85..4d8eec398e 100644
--- a/testsuite/tests/linear/should_fail/all.T
+++ b/testsuite/tests/linear/should_fail/all.T
@@ -39,3 +39,4 @@ test('T18888', normal, compile_fail, [''])
test('T18888_datakinds', normal, compile_fail, [''])
test('T19120', normal, compile_fail, [''])
test('T20083', normal, compile_fail, ['-XLinearTypes'])
+test('T19361', normal, compile_fail, [''])
diff --git a/testsuite/tests/typecheck/should_fail/T15883b.stderr b/testsuite/tests/typecheck/should_fail/T15883b.stderr
index 21b9305315..b3efbc1b41 100644
--- a/testsuite/tests/typecheck/should_fail/T15883b.stderr
+++ b/testsuite/tests/typecheck/should_fail/T15883b.stderr
@@ -1,6 +1,7 @@
-T15883b.hs:14:1:
- Can't make a derived instance of
+
+T15883b.hs:14:1: error:
+ • Can't make a derived instance of
‘Eq (Foo ('BoxedRep 'Lifted))’ with the stock strategy:
- Don't know how to derive ‘Eq’ for type ‘forall a. a’
- In the stand-alone deriving instance for
+ Don't know how to derive ‘Eq’ for type ‘forall (a :: TYPE rep). a’
+ • In the stand-alone deriving instance for
‘Eq (Foo (BoxedRep Lifted))’
diff --git a/testsuite/tests/typecheck/should_fail/T15883c.stderr b/testsuite/tests/typecheck/should_fail/T15883c.stderr
index 60678c4fcb..2aa1049fa5 100644
--- a/testsuite/tests/typecheck/should_fail/T15883c.stderr
+++ b/testsuite/tests/typecheck/should_fail/T15883c.stderr
@@ -1,6 +1,7 @@
-T15883c.hs:14:1:
- Can't make a derived instance of
+
+T15883c.hs:14:1: error:
+ • Can't make a derived instance of
‘Ord (Foo ('BoxedRep 'Lifted))’ with the stock strategy:
- Don't know how to derive ‘Ord’ for type ‘forall a. a’
- In the stand-alone deriving instance for
+ Don't know how to derive ‘Ord’ for type ‘forall (a :: TYPE rep). a’
+ • In the stand-alone deriving instance for
‘Ord (Foo (BoxedRep Lifted))’
diff --git a/testsuite/tests/typecheck/should_fail/T15883d.stderr b/testsuite/tests/typecheck/should_fail/T15883d.stderr
index 162b31072e..96a294bc9e 100644
--- a/testsuite/tests/typecheck/should_fail/T15883d.stderr
+++ b/testsuite/tests/typecheck/should_fail/T15883d.stderr
@@ -1,6 +1,8 @@
-T15883d.hs:14:1:
- Can't make a derived instance of
+
+T15883d.hs:14:1: error:
+ • Can't make a derived instance of
‘Show (Foo ('BoxedRep 'Lifted))’ with the stock strategy:
- Don't know how to derive ‘Show’ for type ‘forall a. a’
- In the stand-alone deriving instance for
+ Don't know how to derive ‘Show’
+ for type ‘forall (a :: TYPE rep). a’
+ • In the stand-alone deriving instance for
‘Show (Foo (BoxedRep Lifted))’
diff --git a/testsuite/tests/typecheck/should_fail/T15883e.stderr b/testsuite/tests/typecheck/should_fail/T15883e.stderr
index a20b3f5d43..c7006fb790 100644
--- a/testsuite/tests/typecheck/should_fail/T15883e.stderr
+++ b/testsuite/tests/typecheck/should_fail/T15883e.stderr
@@ -1,6 +1,8 @@
-T15883e.hs:16:1:
- Can't make a derived instance of
+
+T15883e.hs:16:1: error:
+ • Can't make a derived instance of
‘Data (Foo ('BoxedRep 'Lifted))’ with the stock strategy:
- Don't know how to derive ‘Data’ for type ‘forall a. a’
- In the stand-alone deriving instance for
+ Don't know how to derive ‘Data’
+ for type ‘forall (a :: TYPE rep). a’
+ • In the stand-alone deriving instance for
‘Data (Foo (BoxedRep Lifted))’